3
0
mirror of https://github.com/triqs/dft_tools synced 2025-01-13 06:28:21 +01:00
dft_tools/triqs/gf/local/fourier_real.cpp
Laura Messio 38d89e2d01 work on gf: scalar_valued, ...
- introducing scalar_valued gf
- Change Fourier routines to run on scalar_valued,
and then use those routines to run on matrix_valued.
- Tools for slices of 2 variables functions
2013-07-19 13:41:34 +02:00

129 lines
5.0 KiB
C++

/*******************************************************************************
*
* TRIQS: a Toolbox for Research in Interacting Quantum Systems
*
* Copyright (C) 2011 by M. Ferrero, O. Parcollet
*
* TRIQS 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.
*
* TRIQS 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
* TRIQS. If not, see <http://www.gnu.org/licenses/>.
*
******************************************************************************/
#include "fourier_real.hpp"
#include <fftw3.h>
namespace triqs { namespace gf {
namespace {
double pi = std::acos(-1);
dcomplex I(0,1);
inline dcomplex th_expo(double t, double a ) { return (t > 0 ? -I * exp(-a*t) : ( t < 0 ? 0 : -0.5 * I * exp(-a*t) ) ) ; }
inline dcomplex th_expo_neg(double t, double a ) { return (t < 0 ? I * exp( a*t) : ( t > 0 ? 0 : 0.5 * I * exp( a*t) ) ) ; }
inline dcomplex th_expo_inv(double w, double a ) { return 1./(w+I*a) ; }
inline dcomplex th_expo_neg_inv(double w, double a ) { return 1./(w-I*a) ; }
}
//--------------------------------------------------------------------------------------
void fourier_impl(gf_view<refreq> gw, gf_view<retime> const gt) {
size_t L = gt.mesh().size();
if (gw.mesh().size() != L) TRIQS_RUNTIME_ERROR << "Meshes are different";
double test = std::abs(gt.mesh().delta() * gw.mesh().delta() * L / (2*pi) -1);
if (test > 1.e-10) TRIQS_RUNTIME_ERROR << "Meshes are not compatible";
const double tmin = gt.mesh().x_min();
const double wmin = gw.mesh().x_min();
//a is a number very larger than delta_w and very smaller than wmax-wmin, used in the tail computation
const double a = gw.mesh().delta() * sqrt( double(L) );
auto ta = gt(freq_infty());
tqa::vector<dcomplex> g_in(L), g_out(L);
for (size_t n1=0; n1<gw.data().shape()[1];n1++) {
for (size_t n2=0; n2<gw.data().shape()[2];n2++) {
dcomplex t1 = ta(1)(n1,n2), t2= ta.get_or_zero(2)(n1,n2);
dcomplex a1 = (t1 + I * t2/a )/2., a2 = (t1 - I * t2/a )/2.;
g_in() = 0;
for (auto const & t : gt.mesh()) {
g_in(t.index()) = (gt(t)(n1,n2) - (a1*th_expo(t,a) + a2*th_expo_neg(t,a))) * std::exp(I*t*wmin);
}
details::fourier_base(g_in, g_out, L, true);
for (auto const & w : gw.mesh()) {
gw(w)(n1,n2) = gt.mesh().delta() * std::exp(I*(w-wmin)*tmin) * g_out(w.index())
+ a1*th_expo_inv(w,a) + a2*th_expo_neg_inv(w,a);
}
}
}
// set tail
gw.singularity() = gt.singularity();
}
//---------------------------------------------------------------------------
void inverse_fourier_impl (gf_view<retime> gt, gf_view<refreq> const gw) {
size_t L = gw.mesh().size();
if ( L != gt.mesh().size()) TRIQS_RUNTIME_ERROR << "Meshes are different";
double test = std::abs(gt.mesh().delta() * gw.mesh().delta() * L / (2*pi) -1);
if (test > 1.e-10) TRIQS_RUNTIME_ERROR << "Meshes are not compatible";
const double tmin = gt.mesh().x_min();
const double wmin = gw.mesh().x_min();
//a is a number very larger than delta_w and very smaller than wmax-wmin, used in the tail computation
const double a = gw.mesh().delta() * sqrt( double(L) );
auto ta = gw(freq_infty());
tqa::vector<dcomplex> g_in(L), g_out(L);
for (size_t n1=0; n1<gt.data().shape()[1];n1++) {
for (size_t n2=0; n2<gt.data().shape()[2];n2++) {
dcomplex t1 = ta(1)(n1,n2), t2 = ta.get_or_zero(2)(n1,n2);
dcomplex a1 = (t1 + I * t2/a )/2., a2 = (t1 - I * t2/a )/2.;
g_in() = 0;
for (auto const & w: gw.mesh())
g_in(w.index()) = (gw(w)(n1,n2) - a1*th_expo_inv(w,a) - a2*th_expo_neg_inv(w,a) ) * std::exp(-I*w*tmin);
details::fourier_base(g_in, g_out, L, false);
const double corr = 1.0/(gt.mesh().delta()*L);
for (auto const & t : gt.mesh())
gt(t)(n1,n2) = corr * std::exp(I*wmin*(tmin-t)) *
g_out( t.index() ) + a1 * th_expo(t,a) + a2 * th_expo_neg(t,a) ;
}
}
// set tail
gt.singularity() = gw.singularity();
}
//---------------------------------------------------------------------------
gf_keeper<tags::fourier,retime> lazy_fourier (gf_view<retime> const & g) { return g;}
gf_keeper<tags::fourier,refreq> lazy_inverse_fourier (gf_view<refreq> const & g) { return g;}
void triqs_gf_view_assign_delegation( gf_view<refreq> g, gf_keeper<tags::fourier,retime> const & L) { fourier_impl (g,L.g);}
void triqs_gf_view_assign_delegation( gf_view<retime> g, gf_keeper<tags::fourier,refreq> const & L) { inverse_fourier_impl(g,L.g);}
}}