|  | @@ -0,0 +1,416 @@
 | 
	
		
			
				|  |  | +#ifndef SRC_NMIE_APPLIED_IMPL_HPP_
 | 
	
		
			
				|  |  | +#define SRC_NMIE_APPLIED_IMPL_HPP_
 | 
	
		
			
				|  |  | +///
 | 
	
		
			
				|  |  | +/// @file   nmie-applied-impl.hpp
 | 
	
		
			
				|  |  | +/// @author Ladutenko Konstantin <kostyfisik at gmail (.) com>
 | 
	
		
			
				|  |  | +/// @date   Tue Sep  3 00:38:27 2013
 | 
	
		
			
				|  |  | +/// @copyright 2013-2016 Ladutenko Konstantin
 | 
	
		
			
				|  |  | +///
 | 
	
		
			
				|  |  | +/// nmie is free software: you can redistribute it and/or modify
 | 
	
		
			
				|  |  | +/// it under the terms of the GNU General Public License as published by
 | 
	
		
			
				|  |  | +/// the Free Software Foundation, either version 3 of the License, or
 | 
	
		
			
				|  |  | +/// (at your option) any later version.
 | 
	
		
			
				|  |  | +///
 | 
	
		
			
				|  |  | +/// nmie-wrapper is distributed in the hope that it will be useful,
 | 
	
		
			
				|  |  | +/// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
	
		
			
				|  |  | +/// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
	
		
			
				|  |  | +/// GNU General Public License for more details.
 | 
	
		
			
				|  |  | +///
 | 
	
		
			
				|  |  | +/// You should have received a copy of the GNU General Public License
 | 
	
		
			
				|  |  | +/// along with nmie-wrapper.  If not, see <http://www.gnu.org/licenses/>.
 | 
	
		
			
				|  |  | +///
 | 
	
		
			
				|  |  | +/// nmie uses nmie.c from scattnlay by Ovidio Pena
 | 
	
		
			
				|  |  | +/// <ovidio@bytesfall.com> . He has an additional condition to 
 | 
	
		
			
				|  |  | +/// his library:
 | 
	
		
			
				|  |  | +//    The only additional condition is that we expect that all publications         //
 | 
	
		
			
				|  |  | +//    describing  work using this software , or all commercial products             //
 | 
	
		
			
				|  |  | +//    using it, cite the following reference:                                       //
 | 
	
		
			
				|  |  | +//    [1] O. Pena and U. Pal, "Scattering of electromagnetic radiation by           //
 | 
	
		
			
				|  |  | +//        a multilayered sphere," Computer Physics Communications,                  //
 | 
	
		
			
				|  |  | +//        vol. 180, Nov. 2009, pp. 2348-2354.                                       //
 | 
	
		
			
				|  |  | +///
 | 
	
		
			
				|  |  | +/// @brief  Wrapper class around nMie function for ease of use
 | 
	
		
			
				|  |  | +///
 | 
	
		
			
				|  |  | +#include "nmie-applied.hpp"
 | 
	
		
			
				|  |  | +#include "nmie-precision.hpp"
 | 
	
		
			
				|  |  | +#include <array>
 | 
	
		
			
				|  |  | +#include <algorithm>
 | 
	
		
			
				|  |  | +#include <cstdio>
 | 
	
		
			
				|  |  | +#include <cstdlib>
 | 
	
		
			
				|  |  | +#include <stdexcept>
 | 
	
		
			
				|  |  | +#include <vector>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +namespace nmie {  
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::GetFailed() {
 | 
	
		
			
				|  |  | +    FloatType faild_x = 9.42477796076938;
 | 
	
		
			
				|  |  | +    //FloatType faild_x = 9.42477796076937;
 | 
	
		
			
				|  |  | +    std::complex<FloatType> z(faild_x, 0.0);
 | 
	
		
			
				|  |  | +    std::vector<int> nmax_local_array = {20, 100, 500, 2500};
 | 
	
		
			
				|  |  | +    for (auto nmax_local : nmax_local_array) {
 | 
	
		
			
				|  |  | +      std::vector<std::complex<FloatType> > D1_failed(nmax_local + 1);
 | 
	
		
			
				|  |  | +      // Downward recurrence for D1 - equations (16a) and (16b)
 | 
	
		
			
				|  |  | +      D1_failed[nmax_local] = std::complex<FloatType>(0.0, 0.0);
 | 
	
		
			
				|  |  | +      const std::complex<FloatType> zinv = std::complex<FloatType>(1.0, 0.0)/z;
 | 
	
		
			
				|  |  | +      for (int n = nmax_local; n > 0; n--) {
 | 
	
		
			
				|  |  | +        D1_failed[n - 1] = FloatType(n)*zinv - 1.0/(D1_failed[n] + FloatType(n)*zinv);
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +      printf("Faild D1[0] from reccurence (z = %16.14f, nmax = %d): %g\n",
 | 
	
		
			
				|  |  | +             faild_x, nmax_local, D1_failed[0].real());
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +    printf("Faild D1[0] from continued fraction (z = %16.14f): %g\n", faild_x,
 | 
	
		
			
				|  |  | +           calcD1confra(0,z).real());
 | 
	
		
			
				|  |  | +    //D1[nmax_] = calcD1confra(nmax_, z);
 | 
	
		
			
				|  |  | +  
 | 
	
		
			
				|  |  | +    
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::AddTargetLayer(FloatType width, std::complex<FloatType> layer_index) {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    if (width <= 0)
 | 
	
		
			
				|  |  | +      throw std::invalid_argument("Layer width should be positive!");
 | 
	
		
			
				|  |  | +    target_width_.push_back(width);
 | 
	
		
			
				|  |  | +    target_index_.push_back(layer_index);
 | 
	
		
			
				|  |  | +  }  // end of void  MultiLayerMieApplied<FloatType>::AddTargetLayer(...)  
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::SetTargetPEC(FloatType radius) {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    if (target_width_.size() != 0 || target_index_.size() != 0)
 | 
	
		
			
				|  |  | +      throw std::invalid_argument("Error! Define PEC target radius before any other layers!");
 | 
	
		
			
				|  |  | +    // Add layer of any index...
 | 
	
		
			
				|  |  | +    AddTargetLayer(radius, std::complex<FloatType>(0.0, 0.0));
 | 
	
		
			
				|  |  | +    // ... and mark it as PEC
 | 
	
		
			
				|  |  | +    this->SetPECLayer(0);
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::SetCoatingIndex(std::vector<std::complex<FloatType> > index) {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    coating_index_.clear();
 | 
	
		
			
				|  |  | +    for (auto value : index) coating_index_.push_back(value);
 | 
	
		
			
				|  |  | +  }  // end of void MultiLayerMieApplied<FloatType>::SetCoatingIndex(std::vector<complex> index);  
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::SetCoatingWidth(std::vector<FloatType> width) {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    coating_width_.clear();
 | 
	
		
			
				|  |  | +    for (auto w : width)
 | 
	
		
			
				|  |  | +      if (w <= 0)
 | 
	
		
			
				|  |  | +        throw std::invalid_argument("Coating width should be positive!");
 | 
	
		
			
				|  |  | +      else coating_width_.push_back(w);
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // end of void MultiLayerMieApplied<FloatType>::SetCoatingWidth(...);
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::SetWidthSP(const std::vector<FloatType>& size_parameter) {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    this->size_param_.clear();
 | 
	
		
			
				|  |  | +    FloatType prev_size_parameter = 0.0;
 | 
	
		
			
				|  |  | +    for (auto layer_size_parameter : size_parameter) {
 | 
	
		
			
				|  |  | +      if (layer_size_parameter <= 0.0)
 | 
	
		
			
				|  |  | +        throw std::invalid_argument("Size parameter should be positive!");
 | 
	
		
			
				|  |  | +      if (prev_size_parameter > layer_size_parameter) 
 | 
	
		
			
				|  |  | +        throw std::invalid_argument
 | 
	
		
			
				|  |  | +          ("Size parameter for next layer should be larger than the previous one!");
 | 
	
		
			
				|  |  | +      prev_size_parameter = layer_size_parameter;
 | 
	
		
			
				|  |  | +      this->size_param_.push_back(layer_size_parameter);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // end of void MultiLayerMieApplied<FloatType>::SetWidthSP(...);
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::SetIndexSP(const std::vector< std::complex<FloatType> >& index) {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    //refractive_index_.clear();
 | 
	
		
			
				|  |  | +    this->refractive_index_ = index;
 | 
	
		
			
				|  |  | +    // for (auto value : index) refractive_index_.push_back(value);
 | 
	
		
			
				|  |  | +  }  // end of void MultiLayerMieApplied<FloatType>::SetIndexSP(...);  
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::SetFieldPointsSP(const std::vector< std::vector<FloatType> >& coords_sp) {
 | 
	
		
			
				|  |  | +    if (coords_sp.size() != 3)
 | 
	
		
			
				|  |  | +      throw std::invalid_argument("Error! Wrong dimension of field monitor points!");
 | 
	
		
			
				|  |  | +    if (coords_sp[0].size() != coords_sp[1].size() || coords_sp[0].size() != coords_sp[2].size())
 | 
	
		
			
				|  |  | +      throw std::invalid_argument("Error! Missing coordinates for field monitor points!");
 | 
	
		
			
				|  |  | +    coords_sp_ = coords_sp;
 | 
	
		
			
				|  |  | +    // for (int i = 0; i < coords_sp_[0].size(); ++i) {
 | 
	
		
			
				|  |  | +    //   printf("%g, %g, %g\n", coords_sp_[0][i], coords_sp_[1][i], coords_sp_[2][i]);
 | 
	
		
			
				|  |  | +    // }
 | 
	
		
			
				|  |  | +  }  // end of void MultiLayerMieApplied<FloatType>::SetFieldPointsSP(...)
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::GenerateSizeParameter() {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    this->size_param_.clear();
 | 
	
		
			
				|  |  | +    FloatType radius = 0.0;
 | 
	
		
			
				|  |  | +    for (auto width : target_width_) {
 | 
	
		
			
				|  |  | +      radius += width;
 | 
	
		
			
				|  |  | +      this->size_param_.push_back(2*this->PI_*radius/wavelength_);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +    for (auto width : coating_width_) {
 | 
	
		
			
				|  |  | +      radius += width;
 | 
	
		
			
				|  |  | +      this->size_param_.push_back(2*this->PI_*radius/wavelength_);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +    this->total_radius_ = radius;
 | 
	
		
			
				|  |  | +  }  // end of void MultiLayerMieApplied<FloatType>::GenerateSizeParameter();
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::GenerateIndex() {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    this->refractive_index_.clear();
 | 
	
		
			
				|  |  | +    for (auto index : this->target_index_)
 | 
	
		
			
				|  |  | +      this->refractive_index_.push_back(index);
 | 
	
		
			
				|  |  | +    for (auto index : this->coating_index_)
 | 
	
		
			
				|  |  | +      this->refractive_index_.push_back(index);
 | 
	
		
			
				|  |  | +  }  // end of void MultiLayerMieApplied<FloatType>::GenerateIndex();
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  FloatType MultiLayerMieApplied<FloatType>::GetTotalRadius() {
 | 
	
		
			
				|  |  | +    if (!this->isMieCalculated())  GenerateSizeParameter();
 | 
	
		
			
				|  |  | +    return this->total_radius_;      
 | 
	
		
			
				|  |  | +  }  // end of FloatType MultiLayerMieApplied<FloatType>::GetTotalRadius();
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  std::vector< std::vector<FloatType> >
 | 
	
		
			
				|  |  | +  MultiLayerMieApplied<FloatType>::GetSpectra(FloatType from_WL, FloatType to_WL, int samples) {
 | 
	
		
			
				|  |  | +    if (!this->isMieCalculated())
 | 
	
		
			
				|  |  | +      throw std::invalid_argument("You should run calculations before result request!");
 | 
	
		
			
				|  |  | +    std::vector< std::vector<FloatType> > spectra;
 | 
	
		
			
				|  |  | +    FloatType step_WL = (to_WL - from_WL)/static_cast<FloatType>(samples);
 | 
	
		
			
				|  |  | +    FloatType wavelength_backup = wavelength_;
 | 
	
		
			
				|  |  | +    long fails = 0;
 | 
	
		
			
				|  |  | +    for (FloatType WL = from_WL; WL < to_WL; WL += step_WL) {
 | 
	
		
			
				|  |  | +      wavelength_ = WL;
 | 
	
		
			
				|  |  | +      try {
 | 
	
		
			
				|  |  | +        RunMieCalculation();
 | 
	
		
			
				|  |  | +      } catch(const std::invalid_argument& ia) {
 | 
	
		
			
				|  |  | +        fails++;
 | 
	
		
			
				|  |  | +        continue;
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +      //printf("%3.1f ",WL);
 | 
	
		
			
				|  |  | +      spectra.push_back(std::vector<FloatType>({wavelength_, this->GetQext(),
 | 
	
		
			
				|  |  | +	      this->GetQsca(), this->GetQabs(), this->GetQbk()}));
 | 
	
		
			
				|  |  | +    }  // end of for each WL in spectra
 | 
	
		
			
				|  |  | +    printf("Spectrum has %li fails\n",fails);
 | 
	
		
			
				|  |  | +    wavelength_ = wavelength_backup;
 | 
	
		
			
				|  |  | +    return spectra;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::ClearTarget() {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    this->target_width_.clear();
 | 
	
		
			
				|  |  | +    this->target_index_.clear();
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::ClearCoating() {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    this->coating_width_.clear();
 | 
	
		
			
				|  |  | +    this->coating_index_.clear();
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::ClearLayers() {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    this->ClearTarget();
 | 
	
		
			
				|  |  | +    this->ClearCoating();
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::ClearAllDesign() {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    this->ClearLayers();
 | 
	
		
			
				|  |  | +    this->size_param_.clear();
 | 
	
		
			
				|  |  | +    this->refractive_index_.clear();
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  //                         Computational core
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  //**********************************************************************************//
 | 
	
		
			
				|  |  | +  // Function CONFRA ported from MIEV0.f (Wiscombe,1979)
 | 
	
		
			
				|  |  | +  // Ref. to NCAR Technical Notes, Wiscombe, 1979
 | 
	
		
			
				|  |  | +  /*
 | 
	
		
			
				|  |  | +c         Compute Bessel function ratio A-sub-N from its
 | 
	
		
			
				|  |  | +c         continued fraction using Lentz method
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c         ZINV = Reciprocal of argument of A
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    I N T E R N A L    V A R I A B L E S
 | 
	
		
			
				|  |  | +c    ------------------------------------
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    CAK      Term in continued fraction expansion of A (Eq. R25)
 | 
	
		
			
				|  |  | +c     a_k
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    CAPT     Factor used in Lentz iteration for A (Eq. R27)
 | 
	
		
			
				|  |  | +c     T_k
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    CNUMER   Numerator   in capT  (Eq. R28A)
 | 
	
		
			
				|  |  | +c     N_k
 | 
	
		
			
				|  |  | +c    CDENOM   Denominator in capT  (Eq. R28B)
 | 
	
		
			
				|  |  | +c     D_k
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    CDTD     Product of two successive denominators of capT factors
 | 
	
		
			
				|  |  | +c                 (Eq. R34C)
 | 
	
		
			
				|  |  | +c     xi_1
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    CNTN     Product of two successive numerators of capT factors
 | 
	
		
			
				|  |  | +c                 (Eq. R34B)
 | 
	
		
			
				|  |  | +c     xi_2
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    EPS1     Ill-conditioning criterion
 | 
	
		
			
				|  |  | +c    EPS2     Convergence criterion
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    KK       Subscript k of cAk  (Eq. R25B)
 | 
	
		
			
				|  |  | +c     k
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    KOUNT    Iteration counter (used to prevent infinite looping)
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    MAXIT    Max. allowed no. of iterations
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +c    MM + 1  and - 1, alternately
 | 
	
		
			
				|  |  | +*/
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  std::complex<FloatType> MultiLayerMieApplied<FloatType>::calcD1confra(const int N, const std::complex<FloatType> z) {
 | 
	
		
			
				|  |  | +  // NTMR -> nmax_ - 1  \\TODO nmax_ ?
 | 
	
		
			
				|  |  | +    //int N = nmax_ - 1;
 | 
	
		
			
				|  |  | +    int KK, KOUNT, MAXIT = 10000, MM;
 | 
	
		
			
				|  |  | +    //    FloatType EPS1=1.0e-2;
 | 
	
		
			
				|  |  | +    FloatType EPS2=1.0e-8;
 | 
	
		
			
				|  |  | +    std::complex<FloatType> CAK, CAPT, CDENOM, CDTD, CNTN, CNUMER;
 | 
	
		
			
				|  |  | +    std::complex<FloatType> one = std::complex<FloatType>(1.0,0.0);
 | 
	
		
			
				|  |  | +    std::complex<FloatType> ZINV = one/z;
 | 
	
		
			
				|  |  | +// c                                 ** Eq. R25a
 | 
	
		
			
				|  |  | +    std::complex<FloatType> CONFRA = static_cast<std::complex<FloatType> >(N + 1)*ZINV;   //debug ZINV
 | 
	
		
			
				|  |  | +    MM = - 1; 
 | 
	
		
			
				|  |  | +    KK = 2*N +3; //debug 3
 | 
	
		
			
				|  |  | +// c                                 ** Eq. R25b, k=2
 | 
	
		
			
				|  |  | +    CAK    = static_cast<std::complex<FloatType> >(MM*KK)*ZINV; //debug -3 ZINV
 | 
	
		
			
				|  |  | +    CDENOM = CAK;
 | 
	
		
			
				|  |  | +    CNUMER = CDENOM + one/CONFRA; //-3zinv+z
 | 
	
		
			
				|  |  | +    KOUNT  = 1;
 | 
	
		
			
				|  |  | +    //10 CONTINUE
 | 
	
		
			
				|  |  | +    do {      ++KOUNT;
 | 
	
		
			
				|  |  | +      if (KOUNT > MAXIT) {
 | 
	
		
			
				|  |  | +        printf("re(%g):im(%g)\t\n", CONFRA.real(), CONFRA.imag());
 | 
	
		
			
				|  |  | +        throw std::invalid_argument("ConFra--Iteration failed to converge!\n");
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +      MM *= - 1;      KK += 2;  //debug  mm=1 kk=5
 | 
	
		
			
				|  |  | +      CAK = static_cast<std::complex<FloatType> >(MM*KK)*ZINV; //    ** Eq. R25b //debug 5zinv
 | 
	
		
			
				|  |  | +     //  //c ** Eq. R32    Ill-conditioned case -- stride two terms instead of one
 | 
	
		
			
				|  |  | +     //  if (std::abs(CNUMER/CAK) >= EPS1 ||  std::abs(CDENOM/CAK) >= EPS1) {
 | 
	
		
			
				|  |  | +     //         //c                       ** Eq. R34
 | 
	
		
			
				|  |  | +     //         CNTN   = CAK*CNUMER + 1.0;
 | 
	
		
			
				|  |  | +     //         CDTD   = CAK*CDENOM + 1.0;
 | 
	
		
			
				|  |  | +     //         CONFRA = (CNTN/CDTD)*CONFRA; // ** Eq. R33
 | 
	
		
			
				|  |  | +     //         MM  *= - 1;        KK  += 2;
 | 
	
		
			
				|  |  | +     //         CAK = static_cast<std::complex<FloatType> >(MM*KK)*ZINV; // ** Eq. R25b
 | 
	
		
			
				|  |  | +     //         //c                        ** Eq. R35
 | 
	
		
			
				|  |  | +     //         CNUMER = CAK + CNUMER/CNTN;
 | 
	
		
			
				|  |  | +     //         CDENOM = CAK + CDENOM/CDTD;
 | 
	
		
			
				|  |  | +     //         ++KOUNT;
 | 
	
		
			
				|  |  | +     //         //GO TO  10
 | 
	
		
			
				|  |  | +     //         continue;
 | 
	
		
			
				|  |  | +     // } else { //c                           *** Well-conditioned case
 | 
	
		
			
				|  |  | +      {
 | 
	
		
			
				|  |  | +        CAPT   = CNUMER/CDENOM; // ** Eq. R27 //debug (-3zinv + z)/(-3zinv)
 | 
	
		
			
				|  |  | +        // printf("re(%g):im(%g)**\t", CAPT.real(), CAPT.imag());
 | 
	
		
			
				|  |  | +       CONFRA = CAPT*CONFRA; // ** Eq. R26
 | 
	
		
			
				|  |  | +       //if (N == 0) {output=true;printf(" re:");prn(CONFRA.real());printf(" im:"); prn(CONFRA.imag());output=false;};
 | 
	
		
			
				|  |  | +       //c                                  ** Check for convergence; Eq. R31
 | 
	
		
			
				|  |  | +       if (std::abs(CAPT.real() - 1.0) >= EPS2 ||  std::abs(CAPT.imag()) >= EPS2) {
 | 
	
		
			
				|  |  | +//c                                        ** Eq. R30
 | 
	
		
			
				|  |  | +         CNUMER = CAK + one/CNUMER;
 | 
	
		
			
				|  |  | +         CDENOM = CAK + one/CDENOM;
 | 
	
		
			
				|  |  | +         continue;
 | 
	
		
			
				|  |  | +         //GO TO  10
 | 
	
		
			
				|  |  | +       }  // end of if < eps2
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +      break;
 | 
	
		
			
				|  |  | +    } while(1);    
 | 
	
		
			
				|  |  | +    //if (N == 0)  printf(" return confra for z=(%g,%g)\n", ZINV.real(), ZINV.imag());
 | 
	
		
			
				|  |  | +    return CONFRA;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::ConvertToSP() {
 | 
	
		
			
				|  |  | +    this->MarkUncalculated();
 | 
	
		
			
				|  |  | +    if (target_width_.size() + coating_width_.size() == 0)
 | 
	
		
			
				|  |  | +      return;  // Nothing to convert, we suppose that SP was set directly
 | 
	
		
			
				|  |  | +    GenerateSizeParameter();
 | 
	
		
			
				|  |  | +    GenerateIndex();
 | 
	
		
			
				|  |  | +    if (this->size_param_.size() != this->refractive_index_.size())
 | 
	
		
			
				|  |  | +      throw std::invalid_argument("Ivalid conversion of width to size parameter units!/n");
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::RunMieCalculation() {
 | 
	
		
			
				|  |  | +    ConvertToSP();
 | 
	
		
			
				|  |  | +    this->MultiLayerMie<FloatType>::RunMieCalculation(); 
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  template <typename FloatType>
 | 
	
		
			
				|  |  | +  void MultiLayerMieApplied<FloatType>::GetExpanCoeffs( std::vector< std::vector<std::complex<FloatType> > >& aln, std::vector< std::vector<std::complex<FloatType> > >& bln, std::vector< std::vector<std::complex<FloatType> > >& cln, std::vector< std::vector<std::complex<FloatType> > >& dln) {
 | 
	
		
			
				|  |  | +    ConvertToSP();  // Case of call before running full Mie calculation.
 | 
	
		
			
				|  |  | +    // Calculate scattering coefficients an_ and bn_
 | 
	
		
			
				|  |  | +    this->calcScattCoeffs();
 | 
	
		
			
				|  |  | +    // Calculate expansion coefficients aln_,  bln_, cln_, and dln_
 | 
	
		
			
				|  |  | +    this->calcExpanCoeffs();
 | 
	
		
			
				|  |  | +    aln = this->aln_;
 | 
	
		
			
				|  |  | +    bln = this->bln_;
 | 
	
		
			
				|  |  | +    cln = this->cln_;
 | 
	
		
			
				|  |  | +    dln = this->dln_;
 | 
	
		
			
				|  |  | +    
 | 
	
		
			
				|  |  | +  }  // end of void MultiLayerMieApplied<FloatType>::GetExpanCoeffs( ...)
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +  // ********************************************************************** //
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +}  // end of namespace nmie
 | 
	
		
			
				|  |  | +#endif  // SRC_NMIE_APPLIED_IMPL_HPP_
 |