Skip to content

Commit

Permalink
Merge pull request #17 from mili1247/unstable
Browse files Browse the repository at this point in the history
Add four-point vertex measurement
  • Loading branch information
mili1247 authored Nov 27, 2024
2 parents e55430f + 83360af commit 665e5a7
Show file tree
Hide file tree
Showing 10 changed files with 418 additions and 0 deletions.
1 change: 1 addition & 0 deletions c++/triqs_ctseg/measures.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -25,3 +25,4 @@
#include "./measures/average_sign.hpp"
#include "./measures/pert_order.hpp"
#include "./measures/state_hist.hpp"
#include "./measures/four_point.hpp"
277 changes: 277 additions & 0 deletions c++/triqs_ctseg/measures/four_point.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,277 @@
// Copyright (c) 2024 Simons Foundation
//
// This program 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.
//
// This program 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 may obtain a copy of the License at
// https://www.gnu.org/licenses/gpl-3.0.txt
//
// Authors: Hao Lu

#include "./four_point.hpp"
#include "../logs.hpp"

namespace triqs_ctseg::measures {

four_point::four_point(params_t const &p, work_data_t const &wdata, configuration_t const &config, results_t &results)
: wdata{wdata}, config{config}, results{results} {

beta = p.beta;
measure_g3w = p.measure_g3w;
measure_f3w = p.measure_f3w;

auto g3w_vec = [&]() {
std::vector<gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>>> green_v;
for (auto const &[bl1_name, bl1_size] : wdata.gf_struct)
for (auto const &[bl2_name, bl2_size] : wdata.gf_struct)
green_v.emplace_back(gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>>(
{{beta, Fermion, p.n_w_f_vertex, imfreq::option::all_frequencies},
{beta, Fermion, p.n_w_f_vertex, imfreq::option::all_frequencies},
{beta, Boson, p.n_w_b_vertex, imfreq::option::positive_frequencies_only}},
make_shape(bl1_size, bl1_size, bl2_size, bl2_size)));
return green_v;
};

for (auto const &[bl_name, bl_size] : wdata.gf_struct) block_names.push_back(bl_name);
auto bosonic_block_names = std::vector<std::string>{};
for (auto const &str1 : block_names)
for (auto const &str2 : block_names)
bosonic_block_names.push_back(str1 + "|" + str2);

g3w = make_block_gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>>(bosonic_block_names, g3w_vec());
f3w = make_block_gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>>(bosonic_block_names, g3w_vec());
g3w() = 0;
f3w() = 0;

n_w_fermionic = std::get<0>(g3w[0].mesh().components()).last_index() + 1;
n_w_bosonic = std::get<2>(g3w[0].mesh().components()).last_index() + 1;

w_ini = (2 * (-n_w_fermionic) + 1) * M_PI / beta;
w_inc = 2 * M_PI / beta;

}

// -------------------------------------

void four_point::accumulate(double s) {

LOG("\n ============ MEASURE FOUR-POINT CORRELATION FUNCTION ============ \n");

/// Measure the four-point correlation function
/**
*The four-point correlation function is defined as:
*
*$$\chi^{\sigma\sigma'}_{abcd}(i\omega, i\omega',i\Omega) = G^{2,\sigma,
*\sigma'}_{abcd}(i\omega, i\omega',i\Omega) = \langle c_{a\sigma}(i\omega)
*c^\dagger_{b\sigma}(i\omega+i\Omega) c_{c\sigma'}(i\omega'+i\Omega)
*c^\dagger_{d\sigma'}(i\omega') \rangle$$
*
* Its improved estimator is the Fourier transform of
*
*$$F^{3,\sigma\sigma'}_{abcd}(\tau,\tau',\tau'') = \int \mathrm{d}\bar{\tau}
*\sum_{e\bar{\sigma}} \mathcal{U}^{\sigma\bar{\sigma}}_{ae}(\bar{\tau}-\tau)
*\langle c_{a\sigma}(\tau) c^\dagger_{b\sigma}(\tau') c_{c\sigma'}(\tau'')
*c^\dagger_{d\sigma'}(0) \rangle$$
*
* The vertex corresponding to this correlation function is evaluated separately.
*/

Z += s;

auto const mesh_fermionic = std::get<0>(g3w[0].mesh());
auto const mesh_bosonic = std::get<2>(g3w[0].mesh());

Mw_vector = compute_Mw(false);

if (measure_g3w) {
for (long bl = 0; bl < g3w.size(); bl++) { // bl : 'upup', 'updn', ...
long b1 = bl / wdata.gf_struct.size();
long b2 = bl % wdata.gf_struct.size();
for (int a = 0; a < g3w[bl].target_shape()[0]; a++) {
for (int b = 0; b < g3w[bl].target_shape()[1]; b++) {
for (int c = 0; c < g3w[bl].target_shape()[2]; c++) {
for (int d = 0; d < g3w[bl].target_shape()[3]; d++) {
for (int n1 = -n_w_fermionic; n1 < n_w_fermionic; n1++) {
for (int n4 = -n_w_fermionic; n4 < n_w_fermionic; n4++) {
for (int m = 0; m < n_w_bosonic; m++) {
int n2 = n1 + m;
int n3 = n4 + m;
// This structure is ugly. Need someone who familiar with TRIQS to prune this part.
auto freq_1 = mesh_fermionic[n1 + n_w_fermionic];
auto freq_2 = mesh_fermionic[n4 + n_w_fermionic];
auto freq_3 = mesh_bosonic[m];
g3w[bl][freq_1, freq_2, freq_3](a, b, c, d) += s * Mw(b1, a, b, n1, n2) * Mw(b2, c, d, n3, n4);
if (b1 == b2)
g3w[bl][freq_1, freq_2, freq_3](a, b, c, d) -= s * Mw(b1, a, d, n1, n4) * Mw(b2, c, b, n3, n2);
} // m
} // n4
} // n1
} // d
} // c
} // b
} // a
} // bl
} // measure_g3w

if (measure_f3w) {
nMw_vector = compute_Mw(true);
for (long bl = 0; bl < f3w.size(); bl++) { // bl : 'upup', 'updn', ...
long b1 = bl / wdata.gf_struct.size();
long b2 = bl % wdata.gf_struct.size();
for (int a = 0; a < f3w[bl].target_shape()[0]; a++) {
for (int b = 0; b < f3w[bl].target_shape()[1]; b++) {
for (int c = 0; c < f3w[bl].target_shape()[2]; c++) {
for (int d = 0; d < f3w[bl].target_shape()[3]; d++) {
for (int n1 = -n_w_fermionic; n1 < n_w_fermionic; n1++) {
for (int n4 = -n_w_fermionic; n4 < n_w_fermionic; n4++) {
for (int m = 0; m < n_w_bosonic; m++) {
int n2 = n1 + m;
int n3 = n4 + m;
// Please prune this
auto freq_1 = mesh_fermionic[n1 + n_w_fermionic];
auto freq_2 = mesh_fermionic[n4 + n_w_fermionic];
auto freq_3 = mesh_bosonic[m];
f3w[bl][freq_1, freq_2, freq_3](a, b, c, d) += s * nMw(b1, a, b, n1, n2) * Mw(b2, c, d, n3, n4);
if (b1 == b2)
f3w[bl][freq_1, freq_2, freq_3](a, b, c, d) -= s * nMw(b1, a, d, n1, n4) * Mw(b2, c, b, n3, n2);
} // m
} // n4
} // n1
} // d
} // c
} // b
} // a
} // bl
} // measure_f3w

}

// -------------------------------------

void four_point::collect_results(mpi::communicator const &c) {

Z = mpi::all_reduce(Z, c);
if (measure_g3w) {
g3w = mpi::all_reduce(g3w, c);
g3w = g3w / (Z * beta);

std::vector<std::vector<gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>>>> g3w_vec(wdata.gf_struct.size(), std::vector<gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>>>(wdata.gf_struct.size()));
for (int b1 : range(wdata.gf_struct.size())) {
for (int b2 : range(wdata.gf_struct.size())) {
g3w_vec[b1][b2] = g3w[b1 * wdata.gf_struct.size() + b2];
}
}

auto g3w_block = make_block2_gf(block_names, block_names, g3w_vec);
results.g3w = std::move(g3w_block);
}

if (measure_f3w) {
f3w = mpi::all_reduce(f3w, c);
f3w = f3w / (Z * beta);

std::vector<std::vector<gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>>>> f3w_vec(wdata.gf_struct.size(), std::vector<gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>>>(wdata.gf_struct.size()));
for (int b1 : range(wdata.gf_struct.size())) {
for (int b2 : range(wdata.gf_struct.size())) {
f3w_vec[b1][b2] = f3w[b1 * wdata.gf_struct.size() + b2];
}
}

auto f3w_block = make_block2_gf(block_names, block_names, f3w_vec);
results.f3w = std::move(f3w_block);
}

}

// -------------------------------------

double four_point::fprefactor(long const &block, std::pair<tau_t, long> const &y) {

// Copied from G_F_tau.cpp
int color = wdata.block_to_color(block, y.second);
double I_tau = 0;
for (auto const &[c, sl] : itertools::enumerate(config.seglists)) {
auto ntau = n_tau(y.first, sl); // Density to the right of y.first in sl
if (c != color) I_tau += wdata.U(c, color) * ntau;
if (wdata.has_Dt) {
I_tau -= K_overlap(sl, y.first, false, wdata.Kprime, c, color);
if (c == color) I_tau -= 2 * real(wdata.Kprime(0)(c, c));
}
if (wdata.has_Jperp) {
I_tau -= 4 * real(wdata.Kprime_spin(0)(c, color)) * ntau;
I_tau -= 2 * K_overlap(sl, y.first, false, wdata.Kprime_spin, c, color);
}
}
return I_tau;

}

// -------------------------------------

std::vector<array<dcomplex, 4>> four_point::compute_Mw(bool is_nMw) {

int n_w_aux = 2 * n_w_fermionic + n_w_bosonic > 1 ? 2 * n_w_fermionic + n_w_bosonic - 1 : 0;
std::vector<array<dcomplex, 4>> result;
result.resize(wdata.gf_struct.size());

for (auto const &[bl, bl_pair] : itertools::enumerate(wdata.gf_struct)) {
auto const &[bl_name, bl_size] = bl_pair;
result[bl].resize(make_shape(bl_size, bl_size, n_w_aux, n_w_aux));
result[bl]() = 0;
}

for (auto const &[bl, det] : itertools::enumerate(wdata.dets)) {
long N = det.size();
y_exp_ini.resize(N);
y_exp_inc.resize(N);
x_exp_ini.resize(N);
x_exp_inc.resize(N);
y_inner_index.resize(N);
x_inner_index.resize(N);

for (long id : range(N)) {
auto y = det.get_y(id);
auto x = det.get_x(id);
y_exp_ini(id) = std::exp(dcomplex(0, w_ini * double(std::get<0>(y))));
y_exp_inc(id) = std::exp(dcomplex(0, w_inc * double(std::get<0>(y))));
x_exp_ini(id) = std::exp(dcomplex(0, -w_ini * double(std::get<0>(x))));
x_exp_inc(id) = std::exp(dcomplex(0, -w_inc * double(std::get<0>(x))));
y_inner_index(id) = std::get<1>(y);
x_inner_index(id) = std::get<1>(x);
}

for (long id_y : range(N)) {
auto y = det.get_y(id_y);
int yj = y_inner_index(id_y);
double f_fact = is_nMw ? fprefactor(bl, y) : 1.0;

for (long id_x : range(N)) {
int xi = x_inner_index(id_x);
dcomplex y_exp = y_exp_ini(id_y);
dcomplex x_exp = x_exp_ini(id_x);
auto Minv = det.inverse_matrix(id_y, id_x);

for (int n_1 : range(n_w_aux)) {
for (int n_2 : range(n_w_aux)) {
auto val = Minv * y_exp * x_exp;
result[bl](yj, xi, n_1, n_2) += val * f_fact;
x_exp *= x_exp_inc(id_x);
}
x_exp = x_exp_ini(id_x);
y_exp *= y_exp_inc(id_y);
}
}
}
}
return result;

}

} // namespace triqs_ctseg::measures
64 changes: 64 additions & 0 deletions c++/triqs_ctseg/measures/four_point.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@
// Copyright (c) 2024 Simons Foundation
//
// This program 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.
//
// This program 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 may obtain a copy of the License at
// https://www.gnu.org/licenses/gpl-3.0.txt
//
// Authors: Hao Lu

#pragma once
#include "../configuration.hpp"
#include "../work_data.hpp"
#include "../results.hpp"

namespace triqs_ctseg::measures {

struct four_point {

work_data_t const &wdata;
configuration_t const &config;
results_t &results;
double beta;
double w_ini, w_inc;
bool measure_g3w;
bool measure_f3w;
int n_w_fermionic;
int n_w_bosonic;
std::vector<std::string> block_names;
std::vector<array<dcomplex, 4>> Mw_vector, nMw_vector;
vector<dcomplex> y_exp_ini, y_exp_inc, x_exp_ini, x_exp_inc;
vector<int> y_inner_index, x_inner_index;

block_gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>> g3w;
block_gf<prod<imfreq, imfreq, imfreq>, tensor_valued<4>> f3w;

double Z = 0;

four_point(params_t const &params, work_data_t const &wdata, configuration_t const &config, results_t &results);

void accumulate(double s);
void collect_results(mpi::communicator const &c);
double fprefactor(long const &block, std::pair<tau_t, long> const &y);

std::vector<array<dcomplex, 4>> compute_Mw(bool is_nMw);

dcomplex Mw(long const &block, int const &i, int const &j, int const &n1, int const &n2) {
return Mw_vector[block](i, j, n1 + n_w_fermionic, n2 + n_w_fermionic);
}

dcomplex nMw(long const &block, int const &i, int const &j, int const &n1, int const &n2) {
return nMw_vector[block](i, j, n1 + n_w_fermionic, n2 + n_w_fermionic);
}

};

} // namespace triqs_ctseg::measures
6 changes: 6 additions & 0 deletions c++/triqs_ctseg/params.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -51,6 +51,8 @@ namespace triqs_ctseg {
h5_write(grp, "h_loc0", c.h_loc0);
h5_write(grp, "n_tau_G", c.n_tau_G);
h5_write(grp, "n_tau_chi2", c.n_tau_chi2);
h5_write(grp, "n_w_b_vertex", c.n_w_b_vertex);
h5_write(grp, "n_w_f_vertex", c.n_w_f_vertex);
h5_write(grp, "n_cycles", c.n_cycles);
h5_write(grp, "length_cycle", c.length_cycle);
h5_write(grp, "n_warmup_cycles", c.n_warmup_cycles);
Expand All @@ -77,6 +79,8 @@ namespace triqs_ctseg {
h5_write(grp, "measure_nn_tau", c.measure_nn_tau);
h5_write(grp, "measure_Sperp_tau", c.measure_Sperp_tau);
h5_write(grp, "measure_state_hist", c.measure_state_hist);
h5_write(grp, "measure_g3w", c.measure_g3w);
h5_write(grp, "measure_f3w", c.measure_f3w);
h5_write(grp, "det_init_size", c.det_init_size);
h5_write(grp, "det_n_operations_before_check", c.det_n_operations_before_check);
h5_write(grp, "det_precision_warning", c.det_precision_warning);
Expand Down Expand Up @@ -121,6 +125,8 @@ namespace triqs_ctseg {
h5_read(grp, "measure_nn_tau", c.measure_nn_tau);
h5_read(grp, "measure_Sperp_tau", c.measure_Sperp_tau);
h5_read(grp, "measure_state_hist", c.measure_state_hist);
h5_read(grp, "measure_g3w", c.measure_g3w);
h5_read(grp, "measure_f3w", c.measure_f3w);
h5_read(grp, "det_init_size", c.det_init_size);
h5_read(grp, "det_n_operations_before_check", c.det_n_operations_before_check);
h5_read(grp, "det_precision_warning", c.det_precision_warning);
Expand Down
Loading

0 comments on commit 665e5a7

Please sign in to comment.