Program Listing for File model_instance.hpp

Return to documentation for file (src_ed/model_instance.hpp)

#ifndef model_instance_h
#define model_instance_h
#define LOOK_UP
#ifdef _OPENMP
#include <omp.h>
#endif

#include <fstream>
#include <memory>
#include <deque>
#include "model_instance_base.hpp"
#include "Q_matrix_set.hpp"
#include "continued_fraction_set.hpp"
#include "mcf_set.hpp"
#include "ED_basis.hpp"
#include "binary_state.hpp"

#include "Hamiltonian/Hamiltonian_base.hpp"
#include "Hamiltonian/Hamiltonian_Dense.hpp"
#include "Hamiltonian/Hamiltonian_CSR.hpp"
#include "Hamiltonian/Hamiltonian_OnTheFly.hpp"
#include "Hamiltonian/Hamiltonian_Factorized.hpp"
#include "Hamiltonian/Hamiltonian_Operator.hpp"

#ifdef EIGEN_HAMILTONIAN
#include "Hamiltonian/Hamiltonian_Eigen.hpp"
#endif

extern double max_gap;
extern run_statistics STATS;

void polynomial_fit(
  vector<double> &xa,
  vector<double> &ya,
  const double x,
  double &y,
  double &dy
);




template<typename HilbertField>
struct model_instance : model_instance_base
{
  // members
  size_t look_up_size;
  matrix<HilbertField> tc, tcb, tb;
  matrix<HilbertField> tc_down, tcb_down, tb_down;
  matrix<HilbertField> tcb_nd, tb_nd;
  matrix<HilbertField> tcb_nd_down, tb_nd_down;
  set<shared_ptr<state<HilbertField>>> states;
  deque<pair<Complex, matrix<Complex>>> look_up_table;
  deque<pair<Complex, matrix<Complex>>> look_up_table_down;

  model_instance(size_t _label, shared_ptr<model> _the_model, const map<string,double> _value, const string &_sectors);
  Hamiltonian<HilbertField>* create_hamiltonian(
    shared_ptr<model> the_model,
    const map<string, double> &value,
    sector s
  );

  void build_cf(state<HilbertField> &Omega, bool spin_down);
  void build_qmatrix(state<HilbertField> &Omega, bool spin_down);
  void build_mcf(state<HilbertField> &Omega, bool spin_down);
  void build_mcf_from_qmatrix(state<HilbertField> &Omega, bool spin_down);
  void clear_states();
  void compute_weights();
  void insert_state(shared_ptr<state<HilbertField>> S);
  void set_hopping_matrix(bool spin_down);
  void merge_states() override;
  string GS_string() const;
  void build_bases_and_HS_operators(const sector& GS_sector, bool spin_down);
  double fidelity(model_instance<HilbertField>& label);

  // realization of virtual base class methods
  pair<double, string> low_energy_states() override;
  pair<double, double> cluster_averages(shared_ptr<Hermitian_operator> h) override;
  void Green_function_solve() override;
  pair<double, string> one_body_solve() override;
  matrix<Complex>  Green_function(const Complex &z, bool spin_down, bool blocks) override;
  void Green_function_average() override;
  matrix<Complex>  self_energy(const Complex &z, bool spin_down) override;
  matrix<Complex>  hopping_matrix(bool spin_down) override;
  matrix<Complex>  hopping_matrix_full(bool spin_down, bool diag) override;
  vector<tuple<int,int,double>> interactions() override;
  matrix<Complex>  hybridization_function(Complex w, bool spin_down) override;
  vector<Complex>  susceptibility(shared_ptr<Hermitian_operator> h, const vector<Complex> &w) override;
  vector<pair<double,double>> susceptibility_poles(shared_ptr<Hermitian_operator> h) override;
  void Green_function_density() override;
  void print(ostream& fout) override;
  void write_hdf5(H5::Group& grp) override;
  void read_hdf5(H5::Group& grp)  override;
  double tr_sigma_inf() override;
  pair<vector<double>, vector<complex<double>>> qmatrix(bool spin_down);
  pair<vector<double>, vector<complex<double>>> hybridization(bool spin_down);
  void print_wavefunction(ostream& fout) override;
  pair<matrix<Complex>, vector<uint64_t>>  density_matrix_mixed(vector<int> sites) override;
  pair<matrix<Complex>, vector<uint64_t>>  density_matrix_factorized(vector<int> sites) override;
};



//==============================================================================
// implementation of model_instance



template<typename HilbertField>
model_instance<HilbertField>::model_instance(size_t _label, shared_ptr<model> _the_model, const map<string,double> _value, const string &_sectors)
: model_instance_base(_label, _the_model, _value, _sectors)

{
  if(typeid(HilbertField) == typeid(Complex)) complex_Hilbert = true;
  else complex_Hilbert = false;

  // allocating matrices
  tc.set_size(n_mixed*the_model->n_sites);
  tb.set_size(n_mixed*the_model->n_bath);
  tcb.set_size(n_mixed*the_model->n_sites, n_mixed*the_model->n_bath);
  if(mixing&HS_mixing::up_down){
    tc_down.set_size(n_mixed*the_model->n_sites);
    tb_down.set_size(n_mixed*the_model->n_bath);
    tcb_down.set_size(n_mixed*the_model->n_sites,n_mixed*the_model->n_bath);
  }
  set_hopping_matrix(false);
  look_up_size = global_int("GF_lookup_depth");
}





template<typename HilbertField>
pair<double, double> model_instance<HilbertField>::cluster_averages(shared_ptr<Hermitian_operator> h)
{
  if(value.find(h->name)==value.end()) qcm_ED_throw("operator "+h->name+" is not activated in instance "+to_string(label));
  if(h->target != 1 && gf_read) return {NAN, NAN};
  double ave=NAN, var=NAN;
  if(!is_correlated or gf_read){
    ave = h->average_from_GF(M,false);
    if(mixing&HS_mixing::up_down) ave += h->average_from_GF(M_down,true);
    if(mixing == HS_mixing::normal) ave *= 2.0;
    if(mixing == HS_mixing::anomalous) ave += h->nambu_correction;
    if(mixing == HS_mixing::full) {
      ave += h->nambu_correction_full;
      ave *= 0.5;
    }
  }
  else{
    var = 0.0;
    ave = 0.0;
    for(auto& gs : states) h->expectation_value(*gs, ave, var);
    var -= ave*ave;
  }
  if(h->target == 1){
    ave *= h->norm;
    var *= h->norm*h->norm;
  }
  return {ave, var};
}



template<typename HilbertField>
void model_instance<HilbertField>::clear_states(){
  for(auto& x : states) erase(x->psi);
  gs_solved = false;
}


template<typename HilbertField>
Hamiltonian<HilbertField>* model_instance<HilbertField>::create_hamiltonian(
    shared_ptr<model> the_model,
    const map<string, double> &value,
    sector s
) {
    Hamiltonian<HilbertField> *H = nullptr;
    //enforced Hamiltonian format
    if(the_model->is_factorized) {
        H = new Hamiltonian_Factorized<HilbertField>(the_model, value, s);
    }
    else if (the_model->provide_basis(s)->dim < global_int("max_dim_full")) {
        H = new Hamiltonian_Dense<HilbertField>(the_model, value, s);
    }
    else {
        switch(Hamiltonian_format) {
            //native implementation
            case H_format_csr:
                H = new Hamiltonian_CSR<HilbertField>(the_model, value, s);
                break;
            case H_format_dense:
                H = new Hamiltonian_Dense<HilbertField>(the_model, value, s);
                break;
            case H_format_onthefly:
                H = new Hamiltonian_OnTheFly<HilbertField>(the_model, value, s);
                break;
            case H_format_ops:
                H = new Hamiltonian_Operator<HilbertField>(the_model, value, s);
                break;
            case H_format_factorized:
                break;
            //implementation with dependencies
            case H_format_eigen:
#ifdef EIGEN_HAMILTONIAN
                H = new Hamiltonian_Eigen<HilbertField>(the_model, value, s);
                //std::cout << "Hamiltonian CSR" << std::endl;
                break;
#else
                cout << "Warning! qcm_wed not configured with Eigen! Fallback to native CSR implementation" << endl;
                Hamiltonian_format = H_format_csr;
                H = new Hamiltonian_CSR<HilbertField>(the_model, value, s);
                break;
#endif
        }
    }
    return H;
}


template<typename HilbertField>
pair<double, string> model_instance<HilbertField>::low_energy_states()
{
  if(gs_solved) return {GS_energy, GS_string()};
  gs_solved = true;
  bool is_complex = (typeid(HilbertField) == typeid(Complex));

  if(!is_correlated) one_body_solve(); // computes also M, the average cluster GF

  // computing the cluster averages in the cases "uncorrelated" and "read from file"
  if(!is_correlated or gf_read){ // doing this causes a bug if U=0, in move_submatrix...
    averages.reserve(value.size());
    for(auto& x : value){
      auto X = cluster_averages(the_model->term.at(x.first));
      averages.push_back(tuple<string,double,double>(x.first, X.first, X.second));
    }
    return {GS_energy, GS_string()};
  }

  // Now solving by ED
  // building a set of trial sectors according to the target sector;

  if(global_bool("verb_ED")) cout << "computing low-energy state for cluster instance " << full_name() << endl;

  sector_set = target_sectors;

  GS_energy = 1.0e12; // some large number

  // loop over sectors to find the ground state sector

  for(auto& s:sector_set){
    // the_model->build_HS_operators(s, is_complex);
    Hamiltonian<HilbertField> *H = create_hamiltonian(the_model, value, s);
    if(H->dim == 0) continue;

    vector<shared_ptr<state<HilbertField>>> gs;
    try {
      gs = H->states(GS_energy); // finds the low-energy states for this sector and adds them to the list
    }
    catch (const qcm_error& e) {
      delete H;
      qcm_ED_throw(string(e.what())
                   + "\n  in cluster model '" + the_model->name
                   + "', instance '" + full_name()
                   + "', sector '" + s.name() + "'");
    }
    for(auto& x : gs) states.insert(x);
    delete H;
  }

  compute_weights();

  // eliminate states with small weight
  auto min = global_double("minimum_weight");
  set<shared_ptr<state<HilbertField>>> states_tmp;
  double tot_weight=0.0;
  for(auto &x : states){
    if(x->weight >= min){
      states_tmp.insert(x);
      tot_weight += x->weight;
    }
  }
  states.clear();
  states = states_tmp;
  tot_weight = 1/tot_weight;
  for(auto &x : states) x->weight *= tot_weight;



  if(global_bool("verb_ED")){
    cout << "states that are kept : " << endl;
    for(auto &x : states){
      cout << x->sec << '\t' << x->energy << '\t' << x->weight << endl;
    }
  }

  // sector_set is now restricted to the sectors containing low-energy states
  sector_set.clear();
  for(auto &x : states) sector_set.insert(x->sec);

  // This is one of the places where one switches on the flag sector::up_down in the non-mixed case.
  bool up_down = false;
  for(auto &x : states) if(x->sec.S) up_down = true;
  if(mixing == HS_mixing::normal and up_down) mixing = HS_mixing::up_down;

  // average energy and sectors
  GS_energy = 1.0e18;
  E0 = 0.0;
  for(auto &x : states){
    E0 += x->weight * x->energy;
    if(x->energy < GS_energy) GS_energy = x->energy;
    ostringstream sout;
    sout << *x;
    if(global_bool("verb_ED")) cout << sout.str() << endl;
  }

  // average and variances of operators
  averages.reserve(value.size());
  for(auto& x : value){
    auto X = cluster_averages(the_model->term.at(x.first));
    averages.push_back(tuple<string,double,double>(x.first, X.first, X.second));
  }
  return {GS_energy, GS_string()};
}




template<typename HilbertField>
void model_instance<HilbertField>::compute_weights(){

  double temperature = global_double("temperature");


  if(temperature < MIDDLE_VALUE){
    // looping over states to keep only the lowest-energy states
    double E0 = 1e12;
    for(auto &x : states) if(x->energy < E0) E0 = x->energy;

    // erasing the excited states
    auto it = states.begin();
    while(it != states.end()){
      if((*it)->energy - E0 > SMALL_VALUE){
        ((shared_ptr<state<HilbertField>>)*it).reset();
        states.erase(it++);
      }
      else ++it;
    }

    // all remnaining states are degenerate
    double Z = 1.0/states.size();
    for(auto &x : states){
      x->weight = Z;
    }

    if(states.size()==0) qcm_ED_throw("instance " + full_name() + " has no low-energy state!");
    E0 = (*states.begin())->energy;
    return;
  }


  // removing states whose energies are too high
  auto it = states.begin();
  while(it != states.end()){
    if((*it)->energy - GS_energy > max_gap){
      ((shared_ptr<state<HilbertField>>)*it).reset();
      states.erase(it++);
    }
    else ++it;
  }

  // compute the partition function
  double Z = 0.0;
  for(auto &x : states){
    x->weight = exp(-(x->energy-GS_energy)/temperature);
    Z += x->weight;
  }
  E0 = GS_energy-temperature*log(Z);

  Z = 1.0/Z;
  for(auto &x : states){
    x->weight *= Z;
  }

  if(global_bool("verb_ED")){
    cout << "list of states/sectors conserved (maximum gap = " << max_gap << "):\n";
    for(auto &x : states){
      cout << x->sec << "\tE-E0 = " << x->energy-GS_energy << "\tweight = " << x->weight << endl;
    }
  }
}


template<typename HilbertField>
void model_instance<HilbertField>::set_hopping_matrix(bool spin_down)
{
  if(hopping_solved) return;
  matrix<HilbertField>& my_tc = (spin_down and mixing&HS_mixing::up_down)? tc_down :tc;
  matrix<HilbertField>& my_tcb = (spin_down and mixing&HS_mixing::up_down)? tcb_down :tcb;
  matrix<HilbertField>& my_tb = (spin_down and mixing&HS_mixing::up_down)? tb_down :tb;

  my_tc.zero();
  for(auto& x : value){
    Hermitian_operator& op = *the_model->term[x.first];
    switch(op.target){
      case 1:
        op.set_hopping_matrix(value.at(x.first), my_tc, spin_down, mixing);
        break;
      case 2:
        op.set_hopping_matrix(value.at(x.first), my_tb, spin_down, mixing);
        break;
      case 3:
        op.set_hopping_matrix(value.at(x.first), my_tcb, spin_down, mixing);
        break;
    }
  }

  if(spin_down == false) SEF_bath = 0.0;

  // diagonalize the bath
  if(the_model->n_bath){
    if(spin_down) {
      tb_nd_down = tb_down;
      tcb_nd_down = tcb_down;
    }
    else{
      tb_nd = tb;
      tcb_nd = tcb;
    }
    if(my_tb.is_diagonal() == 0){
      matrix<HilbertField> tcb_tmp(my_tcb);
      vector<double> d(my_tb.r);
      matrix<HilbertField> Sb(my_tb);
      my_tb.eigensystem(d, Sb);
      my_tb.zero();
      my_tb.diagonal(d);
      my_tcb.product(tcb_tmp,Sb);
    }

        // contribution of the  bath to the Potthoff functional
        for(size_t i=0; i<the_model->n_bath; ++i) if(realpart(my_tb(i,i)) < 0) SEF_bath -= realpart(my_tb(i,i));
    }
  if((mixing & HS_mixing::up_down) and spin_down == false) set_hopping_matrix(true);
  else if(mixing==HS_mixing::normal) SEF_bath *= 2.0;
  hopping_solved = true;
}



template<typename HilbertField>
void model_instance<HilbertField>::insert_state(shared_ptr<state<HilbertField>> S){

  // action depends on energy of states compared to previously minimum energy in the set
  if(S->energy < GS_energy){
    GS_energy = S->energy;
    GS_sector = S->sec;
    // removing states that have too high an energy
    auto it = states.begin();
    while(it != states.end()){
      if((*it)->energy - GS_energy > max_gap){
        ((shared_ptr<state<HilbertField>>)*it).reset();
        states.erase(it++);
      }
      else ++it;
    }
    states.insert(S);
  }
  else if(S->energy - GS_energy > max_gap) {
    return;
  }
  else{
    states.insert(S);
  }
}






template<typename HilbertField>
void model_instance<HilbertField>::Green_function_solve()
{
  if(gf_solved or gf_read) return;
  if(!is_correlated){
    one_body_solve();
    return;
  }
  if(!gs_solved) low_energy_states();

  if(global_bool("verb_ED")) cout << "GF solver for cluster instance " << full_name() << endl;

  for(auto& x : states){

    bool is_complex = (typeid(HilbertField) == typeid(Complex));

    if(GF_solver == GF_format_CF) build_cf(*x, false);
    else if(GF_solver == GF_format_MCF) build_mcf(*x, false);
    else if(GF_solver == GF_format_Q_to_MCF) build_mcf_from_qmatrix(*x, false);
    else build_qmatrix(*x,false);

    if(mixing&HS_mixing::up_down){
      if(GF_solver == GF_format_CF) build_cf(*x,true);
      else if(GF_solver == GF_format_MCF) build_mcf(*x,true);
      else if(GF_solver == GF_format_Q_to_MCF) build_mcf_from_qmatrix(*x,true);
      else build_qmatrix(*x,true);
    }
  }

  if(states.size() > 1 and GF_solver == GF_format_BL and global_bool("merge_states")){
    cout << states.size() << " states were found. Merging..." << endl;
    merge_states();
  }

  gf_solved = true;
  M.set_size(dim_GF);
  if(mixing&HS_mixing::up_down) M_down.set_size(dim_GF);
  Green_function_average();
  Green_function_density();
}






template<typename HilbertField>
matrix<complex<double>> model_instance<HilbertField>::Green_function(const Complex &z, bool spin_down, bool blocks)
{
#ifdef LOOK_UP
  // first search the look_up table
  auto LKUP = &look_up_table;
  if(spin_down) LKUP = &look_up_table_down;
  matrix<Complex> G_recycled;
  #pragma omp critical (looking_up)
  for(auto &x: *LKUP){
    if(abs(z - x.first) < MIDDLE_VALUE){
      // cout << "recycling G for z = " << z << " and x.first = " << x.first << "  G(0,0) = " << x.second(0,0) << endl;
      G_recycled = x.second;
      //break;
    }
  }
  if(G_recycled.size() > 0){
    STATS.n_GF_recycled += 1;
    return G_recycled;
  }
#endif
  STATS.n_GF_computed += 1;

  //#pragma omp master
  {
    if(spin_down and !(mixing&HS_mixing::up_down or mixing==0))
      qcm_ED_throw("spin_down=True impossible with Hilbert space mixing "+to_string(mixing));
    if(!gf_solved) Green_function_solve();
  }
  block_matrix<Complex> gf_block_matrix(the_model->group->site_irrep_dim*n_mixed);
  if(spin_down and mixing&HS_mixing::up_down){
    for(auto& x : states) x->gf_down->Green_function(z, gf_block_matrix);
  }
  else{
    for(auto& x : states) x->gf->Green_function(z, gf_block_matrix);
  }

  matrix<Complex> G(gf_block_matrix.r);

  if((mixing & HS_mixing::anomalous) && global_bool("strip_anomalous_self")){
    gf_block_matrix.inverse();
    the_model->group->to_site_basis(gf_block_matrix, G, n_mixed);
    auto H = hybridization_function(z, spin_down).v;
    if(the_model->n_bath) G.v += H;
    int R = G.r/2;
    for(int r = R; r<G.r; r++){
      for(int c = 0; c<R; c++){
        G(r,c) = 0.0;
        G(c,r) = 0.0;
      }
    }
    if(the_model->n_bath) G.v -= H;
    G.inverse();
  }
  else if(blocks){
    G = gf_block_matrix.build_matrix();
  }
  else{
    the_model->group->to_site_basis(gf_block_matrix, G, n_mixed);
  }

#ifdef LOOK_UP
  #pragma omp critical (looking_up)
  {
    pair<Complex, matrix<Complex>> P(z,G);
    LKUP->push_front(P);
    if(LKUP->size() > look_up_size) LKUP->pop_back();
  }
#endif
  return G;
}



template<typename HilbertField>
void model_instance<HilbertField>::Green_function_average()
{
  // if(!gf_solved) Green_function_solve();
  block_matrix<Complex> G(the_model->group->site_irrep_dim*n_mixed);
  for(auto& x : states) x->gf->integrated_Green_function(G);
  the_model->group->to_site_basis(G, M, n_mixed);
  if(mixing&HS_mixing::up_down){
    block_matrix<Complex> G_down(the_model->group->site_irrep_dim*n_mixed);
    for(auto& x : states) x->gf_down->integrated_Green_function(G_down);
    the_model->group->to_site_basis(G_down, M_down, n_mixed);
  }
}



template<typename HilbertField>
void model_instance<HilbertField>::Green_function_density()
{
  int I = the_model->n_sites;
  double nG = 0.0;
  for(int i=0; i<I; i++) nG += M(i,i).real();
  if(mixing&HS_mixing::spin_flip) for(int i=0; i<I; i++) nG += M(i+I,i+I).real();
  else if(mixing&HS_mixing::anomalous) for(int i=0; i<I; i++) nG += 1-M(i+I,i+I).real();

  if(mixing&HS_mixing::up_down){
    for(int i=0; i<I; i++) nG += M_down(i,i).real();
  }
  GF_density = nG/the_model->n_sites;
  if(mixing == HS_mixing::normal) GF_density *= 2;
}



template<typename HilbertField>
matrix<Complex> model_instance<HilbertField>::self_energy(const Complex &z, bool spin_down)
{
  block_matrix<Complex> gf_block_matrix(the_model->group->site_irrep_dim*n_mixed);
  if(spin_down and mixing&HS_mixing::up_down){
    for(auto& x : states) x->gf_down->Green_function(z, gf_block_matrix);
  }
  else{
    for(auto& x : states) x->gf->Green_function(z, gf_block_matrix);
  }
  gf_block_matrix.inverse();
  matrix<Complex> S(gf_block_matrix.r);
  the_model->group->to_site_basis(gf_block_matrix, S, n_mixed);
  S += (spin_down and mixing&HS_mixing::up_down) ? tc_down : tc;
  if(the_model->n_bath) S.v += hybridization_function(z, spin_down).v;
  S.v *= -1;
  S += z;
  return S;
}



template<typename HilbertField>
void model_instance<HilbertField>::build_qmatrix(state<HilbertField> &Omega, bool spin_down)
{
  auto& sym_orb = the_model->sym_orb[mixing];
  Q_matrix_set<HilbertField> Qm(the_model->group, mixing);
  Q_matrix_set<HilbertField> Qp(the_model->group, mixing);

  check_signals();
  if(global_bool("verb_ED")){
    cout << "\ncomputing Q-matrix for state of energy " << Omega.energy << " in sector " << Omega.sec.name() << endl;
    if(spin_down) cout << "spin down part" << endl;
  }

  // building the Q matrices
  int ns = 2*sym_orb.size();
  if(global_bool("parallel_sectors")){
    if(global_bool("verb_ED")) cout << "openMP parallelization of Green function computation..." << endl;
    #pragma omp parallel for
    for(int s=0; s< ns; s++){
      int r = s/2;
      int pm = 2*(s%2)-1;

      if(sym_orb[r].size() == 0) continue; // irrep not present
      int spin = (spin_down)? 1:sym_orb[r][0].spin;
      sector target_sec = the_model->group->shift_sector(Omega.sec, pm, spin, r);
      if(!the_model->group->sector_is_valid(target_sec)) continue; // target sector is null
      vector<vector<HilbertField>> phi(sym_orb[r].size());
      bool skip_sector=false;
      for(size_t i=0; i< sym_orb[r].size(); i++){
        symmetric_orbital sorb = sym_orb[r][i];
        if(spin_down) sorb.spin =1;
        if(!the_model->create_or_destroy(pm, sorb, Omega, phi[i], HilbertField(1.0))) skip_sector = true;
      }
      if(skip_sector) continue;
      // Assembling the Hamiltonian and band Lanczos procedure
      Hamiltonian<HilbertField> *H = create_hamiltonian(the_model, value, target_sec);
      if(H->dim==0) continue;

      Q_matrix<HilbertField> Qtmp;
      Qtmp = H->build_Q_matrix(phi);

      delete H; //delete Hamiltonian to prevent memory leak
      H = nullptr;

      Qtmp.e -= Omega.energy; // adjust the eigenvalues by adding/subtracting the GS energy
      if(pm == -1){
        Qtmp.e *= -1.0;
      }
      Qtmp.streamline();
      if(pm==-1)
        Qm.q[r] = Qtmp;
      else{
        Qp.q[r] = Qtmp;
        Qp.q[r].v.cconjugate(); // IMPORTANT. Source of bug found 2021-08-14
      }
    }
  }
  else{
    for(int s=0; s< ns; s++){
      int r = s/2;
      int pm = 2*(s%2)-1;

      if(sym_orb[r].size() == 0) continue; // irrep not present
      int spin = (spin_down)? 1:sym_orb[r][0].spin;
      sector target_sec = the_model->group->shift_sector(Omega.sec, pm, spin, r);
      if(!the_model->group->sector_is_valid(target_sec)) continue; // target sector is null
      vector<vector<HilbertField>> phi(sym_orb[r].size());
      bool skip_sector=false;
      for(size_t i=0; i< sym_orb[r].size(); i++){
        symmetric_orbital sorb = sym_orb[r][i];
        if(spin_down) sorb.spin =1;
        if(!the_model->create_or_destroy(pm, sorb, Omega, phi[i], HilbertField(1.0))) skip_sector = true;
      }
      if(skip_sector) continue;
      // Assembling the Hamiltonian and band Lanczos procedure
      Hamiltonian<HilbertField> *H = create_hamiltonian(the_model, value, target_sec);
      if(H->dim==0) continue;

      Q_matrix<HilbertField> Qtmp;
      Qtmp = H->build_Q_matrix(phi);

      delete H; //delete Hamiltonian to prevent memory leak
      H = nullptr;

      Qtmp.e -= Omega.energy; // adjust the eigenvalues by adding/subtracting the GS energy
      if(pm == -1){
        Qtmp.e *= -1.0;
      }
      Qtmp.streamline();
      if(pm==-1)
        Qm.q[r] = Qtmp;
      else{
        Qp.q[r] = Qtmp;
        Qp.q[r].v.cconjugate(); // IMPORTANT. Source of bug found 2021-08-14
      }
    }
  }

  auto Q = make_shared<Q_matrix_set<HilbertField>>(the_model->group, mixing);
  if(spin_down) Omega.gf_down = Q;
  else Omega.gf = Q;
  for(size_t r=0; r<sym_orb.size(); r++){
    Q->q[r].append(Qm.q[r]);
    Q->q[r].append(Qp.q[r]);
    Q->q[r].check_norm(global_double("accur_Q_matrix"));
    Q->q[r].v.v *= sqrt(Omega.weight);
  }
}




template<typename HilbertField>
void model_instance<HilbertField>::build_mcf(state<HilbertField> &Omega, bool spin_down)
{
  auto& sym_orb = the_model->sym_orb[mixing];

  auto mcf = make_shared<mcf_set<HilbertField>>(the_model->group, mixing);
  if(spin_down) Omega.gf_down = mcf;
  else          Omega.gf      = mcf;

  check_signals();
  if(global_bool("verb_ED")){
    cout << "\ncomputing MCF for state of energy " << Omega.energy
         << " in sector " << Omega.sec.name() << endl;
    if(spin_down) cout << "spin down part" << endl;
  }

  double accur_deflation = global_double("accur_deflation");

  int ns = 2*(int)sym_orb.size();
  for(int s = 0; s < ns; ++s){
    int r  = s / 2;
    int pm = 2*(s % 2) - 1;          // -1 (hole) or +1 (electron)

    if(sym_orb[r].size() == 0) continue;
    int spin = (spin_down) ? 1 : sym_orb[r][0].spin;
    sector target_sec = the_model->group->shift_sector(Omega.sec, pm, spin, r);
    if(!the_model->group->sector_is_valid(target_sec)) continue;

    int p = (int)sym_orb[r].size();  // block size

    // Build starting block: phi[i] = c_i^{pm} |Omega>
    vector<vector<HilbertField>> phi(p);
    bool skip_sector = false;
    for(int i = 0; i < p; ++i){
      symmetric_orbital sorb = sym_orb[r][i];
      if(spin_down) sorb.spin = 1;
      if(!the_model->create_or_destroy(pm, sorb, Omega, phi[i], HilbertField(1.0)))
        skip_sector = true;
    }
    if(skip_sector) continue;

    // Extract the upper-triangular QR factor W of the phi block via modified
    // Gram-Schmidt.  phi[l] = sum_{k<=l} q[k] * W(k,l).
    // If some phi[l] deflates (norm < accur_deflation) we stop at that column
    // and proceed with p_actual < p non-deflated vectors.  The non-square weight
    // W_mcf (p_actual × p) maps the reduced Lanczos block back to the full
    // p-dimensional orbital space so that evaluate() still returns a p×p matrix.
    matrix<HilbertField> W(p, p);   // full QR factor (upper-triangular, p×p)
    int p_actual = p;
    vector<vector<HilbertField>> q(phi);   // orthonormalized starting block
    for(int l = 0; l < p; ++l){
      for(int k = 0; k < l; ++k){
        HilbertField z = q[k] * q[l];    // <q[k] | q[l]>
        W(k, l) = z;
        mult_add(-z, q[k], q[l]);
      }
      double nrm = norm(q[l]);
      if(nrm < accur_deflation){ p_actual = l; break; }  // deflation: stop here
      W(l, l) = HilbertField(nrm);
      q[l] *= 1.0 / nrm;
    }
    if(p_actual == 0) continue;

    // Fold in the state weight so that the spectral weight is Omega.weight.
    W.v *= sqrt(Omega.weight);

    // Build the (possibly non-square) p_actual × p weight matrix for the MCF.
    // If p_actual == p this is just the upper-left p×p block (= W itself).
    matrix<HilbertField> W_mcf(p_actual, p);
    for(int k = 0; k < p_actual; ++k)
      for(int l = 0; l < p; ++l)
        W_mcf(k, l) = W(k, l);

    // Trim the starting block to the p_actual non-deflated vectors.
    q.resize(p_actual);

    // Assemble the Hamiltonian in the target sector
    Hamiltonian<HilbertField> *H = create_hamiltonian(the_model, value, target_sec);
    if(H->dim == 0){ delete H; continue; }

    // Block Lanczos on the p_actual orthonormal starting vectors.
    // block_Lanczos_QR=true (default) uses QR upper-triangular B; false uses polar (Hermitian B).
    vector<matrix<HilbertField>> A, B;
    int M0 = (int)(14 * p_actual * log(1.0 * H->dim));
    if(global_bool("block_Lanczos_QR"))
      blockLanczos(*H, q, A, B, M0, global_bool("verb_ED"));
    else
      blockLanczosSVD(*H, q, A, B, M0, global_bool("verb_ED"));

    delete H; H = nullptr;

    if(A.empty()) continue;

    // Build the matrix continued fraction with energy shift and non-square W_mcf.
    matrix_continued_fraction<HilbertField> frac(A, B, Omega.energy, W_mcf, pm == 1);

    if(pm == -1) mcf->h[r] = frac;
    else         mcf->e[r] = frac;
  }
  mcf->build_combined();
}


template<typename HilbertField>
void model_instance<HilbertField>::build_mcf_from_qmatrix(state<HilbertField> &Omega, bool spin_down)
{
  // Step 1: build the Q_matrix_set using the standard Lehmann path
  build_qmatrix(Omega, spin_down);

  auto Qset = dynamic_pointer_cast<Q_matrix_set<HilbertField>>(
      spin_down ? Omega.gf_down : Omega.gf);
  if(!Qset) return;

  // Step 2: create mcf_set; convert each Q_matrix block to a combined MCF.
  // The Q_matrix already merges electron and hole poles, so each block maps
  // directly to a single combined MCF stored in combined[r].
  auto mcf = make_shared<mcf_set<HilbertField>>(the_model->group, mixing);

  auto& sym_orb = the_model->sym_orb[mixing];
  for(size_t r = 0; r < sym_orb.size(); ++r){
    if(Qset->q[r].M == 0) continue;
    mcf->combined[r] = Q_matrix_to_mcf(Qset->q[r]);
  }

  // Step 3: replace the Q_matrix_set with the new mcf_set
  if(spin_down) Omega.gf_down = mcf;
  else          Omega.gf      = mcf;
}







template<typename HilbertField>
matrix<Complex> model_instance<HilbertField>::hybridization_function(Complex w, bool spin_down){
  set_hopping_matrix(false);
  size_t n_bathg = tb.r;
  matrix<HilbertField>& my_tb = (spin_down and mixing&HS_mixing::up_down)? tb_down : tb;
  matrix<HilbertField>& my_tcb = (spin_down and mixing&HS_mixing::up_down)? tcb_down : tcb;

  matrix<Complex> Gamma(dim_GF);
  for(size_t i=0; i<n_bathg; ++i){
    Complex z = 1.0/(w - my_tb(i,i));
    for(size_t a=0; a < Gamma.r; ++a){
      for(size_t b=0; b < Gamma.r; ++b) Gamma(a,b) +=  my_tcb(a,i) * conjugate(my_tcb(b,i)) * z;
    }
  }
  return Gamma;
}



template<typename HilbertField>
vector<Complex> model_instance<HilbertField>::susceptibility(shared_ptr<Hermitian_operator> h, const vector<Complex> &w){

  if(value.find(h->name)==value.end()) qcm_ED_throw("operator "+h->name+" is not activated in instance "+to_string(label));
  if(!gs_solved) low_energy_states();

  vector<Complex> chi(w.size(),0.0);

  for(auto& sec : sector_set){
    Hamiltonian<HilbertField> *H = create_hamiltonian(the_model, value, sec);
    if(H->dim==0) continue;
    for(auto& gs : states){
      if(gs->sec != sec) continue;
      vector<vector<HilbertField>> b(1);
      b[0].resize(H->dim);
      shared_ptr<HS_Hermitian_operator> hs_op;
      {
        std::lock_guard<std::mutex> lock(h->hs_op_mutex);
        hs_op = h->HS_operator.at(sec);
      }
      hs_op->multiply_add(gs->psi, b[0], 1.0);
      Q_matrix<HilbertField> Q = H->build_Q_matrix(b);
      Q.e -= gs->energy;    // adjust the eigenvalues by subtracting the GS energy
      matrix<Complex> g(1);
      for(size_t j=0; j<w.size(); j++){
        g.zero();
        // Q.Green_function(Complex(-w[j].real(),w[j].imag()),g);
        // g.v *= -1.0;
        Q.Green_function(w[j],g);
        chi[j] += gs->weight*g(0,0);
      }
    }
    delete H; H=nullptr;
  }
  return chi;
}



template<typename HilbertField>
vector<pair<double,double>> model_instance<HilbertField>::susceptibility_poles(shared_ptr<Hermitian_operator> h){

  if(value.find(h->name)==value.end()) qcm_ED_throw("operator "+h->name+" is not activated in instance "+to_string(label));
  if(!gs_solved) low_energy_states();

  vector<pair<double,double>> chi;
  chi.reserve(20);

  for(auto& sec : sector_set){
    Hamiltonian<HilbertField> *H = create_hamiltonian(the_model, value, sec);
    if(H->dim==0) continue;
    for(auto& gs : states){
      if(gs->sec != sec) continue;
      vector<vector<HilbertField>> b(1);
      b[0].resize(H->dim);
      shared_ptr<HS_Hermitian_operator> hs_op;
      {
        std::lock_guard<std::mutex> lock(h->hs_op_mutex);
        hs_op = h->HS_operator.at(sec);
      }
      hs_op->multiply_add(gs->psi, b[0], 1.0);
      Q_matrix<HilbertField> Q = H->build_Q_matrix(b);
      Q.e -= gs->energy;    // adjust the eigenvalues by subtracting the GS energy
      for(size_t i = 0; i<Q.M; i++){
        if(Q.e[i] < 1e-8) continue;
        double r = abs(Q.v(0,i));
        r *= r*gs->weight;
        if(r > 1e-6) chi.push_back({Q.e[i],r});
      }
    }
    delete H; H=nullptr;
  }
  // consolidation
  vector<pair<double,double>> chi2;
  chi2.reserve(20);
  for(size_t i = 0; i< chi.size(); i++){
    size_t j;
    for(j = 0; j< chi2.size(); j++){
      if(fabs(chi2[j].first - chi[i].first) < 1e-4){
        chi2[j].second += chi[i].second; break;
      }
    }
    if(j==chi2.size()) chi2.push_back(chi[i]);
  }
  return chi2;
}




template<typename HilbertField>
void model_instance<HilbertField>::print(ostream& fout)
{
  fout << "\n\nmodel instance no " << label << " based on " << the_model->name;
  if(complex_Hilbert) fout << "  (complex)";
  fout << endl;
  fout << "Symmetric orbitals:\n";
  size_t i = 1;
  for(auto& x : the_model->sym_orb[mixing]){
    for(auto& y : x) {fout << i << '\t' << y << endl; i++;}
  }
  fout << endl;
  for(auto& x: value) fout << x.first << " :\t" << x.second << endl;

  set_hopping_matrix(false);
  fout << "\nhopping matrix:\n" << tc << endl;
  if(the_model->n_bath){
    fout << "\nbath hopping matrix:\n" << tb << endl;
    fout << "\nbath hybridization matrix:\n" << tcb << endl;
    fout << "\nbath hopping matrix (non diagonalized):\n" << tb_nd << endl;
    fout << "\nbath hybridization matrix (non diagonalized):\n" << tcb_nd << endl;
  }
  if(mixing&HS_mixing::up_down){
    fout << "\nhopping matrix (spin down):\n" << tc_down << endl;
    if(the_model->n_bath){
      fout << "\nbath hopping matrix (spin down):\n" << tb_down << endl;
      fout << "\nbath hybridization matrix (spin down):\n" << tcb_down << endl;
      fout << "\nbath hopping matrix (spin_down, non diagonalized):\n" << tb_nd_down << endl;
      fout << "\nbath hybridization matrix (spin_down, non diagonalized):\n" << tcb_nd_down << endl;
    }
  }
}




template<typename HilbertField>
pair<double, string> model_instance<HilbertField>::one_body_solve()
{
  if(the_model->group->g != 1) qcm_ED_throw("The symmetry group must be trivial when using 'one_body_solve()'");

  GS_energy = 0.0;
  set_hopping_matrix(false);
  GF_solver = GF_format_BL;
  auto S = make_shared<state<HilbertField>>();
  auto Qset = make_shared<Q_matrix_set<HilbertField>>(the_model->group, mixing);
  S->gf =  Qset;
  int d = tc.r+tb.r;
  Q_matrix<HilbertField> Q(tc.r, d);
  matrix<HilbertField> H(d);
  tc.move_sub_matrix(tc.r,tc.r, 0, 0, 0, 0, H);
  if(tb.r>0){
    tb.move_sub_matrix(tb.r,tb.r, 0, 0, tc.r, tc.r, H);
    tcb.move_sub_matrix(tcb.r,tcb.c, 0, 0, 0, tc.r, H);
    tcb.move_sub_matrix_HC(tcb.r,tcb.c, 0, 0, tc.r, 0, H);
  }

  matrix<HilbertField> U(d);
  H.eigensystem(Q.e, U);
  U.move_sub_matrix(tc.r, d, 0, 0, 0, 0, Q.v);
  Q.v.cconjugate();
  Qset->q[0] = Q;
  for(auto& x:Q.e) if(x < 0.0) GS_energy += x;


  // computing M, the average M_{ab} = \L c^\dagger_b c_a \R
  M.set_size(d);
  for(int a=0; a<d; a++){
    for(int b=0; b<d; b++){
      for(int r=0; r<Q.e.size(); r++){
        if(Q.e[r] < 0.0) M(b,a) += U(a,r)*conjugate(U(b,r));
      }
    }
  }
  Q.check_norm(1e-6);

  if(mixing&HS_mixing::up_down){
    auto Qset_down = make_shared<Q_matrix_set<HilbertField>>(the_model->group, mixing);
    S->gf_down =  Qset_down;
    Q_matrix<HilbertField> Q_down(tc.r, d);
    matrix<HilbertField> H_down(d);
    tc_down.move_sub_matrix(tc.r,tc.r, 0, 0, 0, 0, H_down);
    if(tb.r>0){
      tb_down.move_sub_matrix(tb.r,tb.r, 0, 0, tc.r, tc.r, H_down);
      tcb_down.move_sub_matrix(tcb.r,tcb.c, 0, 0, 0, tc.r, H_down);
      tcb_down.move_sub_matrix_HC(tcb.r,tcb.c, 0, 0, tc.r, 0, H_down);
    }
    matrix<HilbertField> U(d);
    H_down.eigensystem(Q_down.e, U);
    U.move_sub_matrix(tc.r, d, 0, 0, 0, 0, Q_down.v);
    Q_down.v.cconjugate();
    Qset_down->q[0] = Q_down;
    for(auto& x:Q.e) if(x < 0.0) GS_energy += x;

    // computing M_down, the average M_{ab} = \L c^\dagger_b c_a \R
    M_down.set_size(d);
    for(int a=0; a<d; a++){
      for(int b=0; b<d; b++){
        for(int r=0; r<Q.e.size(); r++){
          if(Q_down.e[r] < 0.0) M_down(b,a) += U(a,r)*conjugate(U(b,r));
        }
      }
    }
  }
  Green_function_density();


  // Nambu correction to the ground state energy
  double nambu_corr = 0.0;
  if(mixing == HS_mixing::anomalous){
    for(auto& x : value) nambu_corr += the_model->term.at(x.first)->nambu_correction * x.second;
  }
  else if(mixing == HS_mixing::full){
    for(auto& x : value) nambu_corr += the_model->term.at(x.first)->nambu_correction_full * x.second;
  }


  GS_energy += nambu_corr;
  if(mixing == HS_mixing::normal) GS_energy *= 2.0;
  else if(mixing == HS_mixing::full) GS_energy *= 0.5;
  S->energy = GS_energy;
  S->weight = 1.0;
  E0 = GS_energy;

  states.insert(S);
  gf_solved = true;
  gs_solved = true;

  return {GS_energy, "uncorrelated"};
}



template<typename HilbertField>
double model_instance<HilbertField>::fidelity(model_instance<HilbertField>& inst)
{
  low_energy_states();
  inst.low_energy_states();
  double z(0);
  for(auto& S1 : states){
    for(auto& S2 : inst.states){
      if(S1->sec != S2->sec) continue;
      HilbertField zz = S1->psi * S2->psi;
      z += abs(zz)*abs(zz) * S1->weight * S2->weight;
    }
  }
  return z;
}


#define NW 3
template<typename HilbertField>
double model_instance<HilbertField>::tr_sigma_inf()
{
  int d1 = dim_GF;
  int d2 = d1;
  if(mixing&1) d2 /= 2;

  vector<double> iw(NW);
  vector<double> S(NW);

  for(int i=0; i<iw.size(); i++) iw[i] = (i+1)*1.0e-5;
  for(int i=0; i<iw.size(); i++){
    complex<double> z(0.0);
    auto sigma = self_energy(complex<double>(0.0, 1.0/iw[i]), false);
    for(int j=0; j<d2; j++) z += sigma(j,j);
    if(mixing&1) for(int j=d2; j<d1; j++) z -= sigma(j,j);
    S[i] = real<double>(z);
  }
  if(mixing&HS_mixing::up_down){
    for(int i=0; i<iw.size(); i++){
      complex<double> z(0.0);
      auto sigma = self_energy(complex<double>(0.0, 1.0/iw[i]), true);
      for(int j=0; j<d2; j++) z += sigma(j,j);
      S[i] += real<double>(z);
    }
  }
  if(mixing == HS_mixing::normal){
    for(int i=0; i<iw.size(); i++) S[i] *= 2;
  }

  // extrapolation to infinite frequency
  double Sinf, Sinf_err;
  polynomial_fit(iw, S, 0.0, Sinf, Sinf_err);
  return Sinf;
}




template<typename HilbertField>
string model_instance<HilbertField>::GS_string() const
{
  if(!is_correlated) return "uncorrelated";

  map<sector, double> ws;
  for(auto& s : states){
    if(ws.find(s->sec) == ws.end()) ws[s->sec] = s->weight;
    else ws[s->sec] += s->weight;
  }
  ostringstream sout;
  for(auto& x: ws){
    sout << x.first << ':' << setprecision(3) << x.second  << '/';
  }
  string out = sout.str();
  out.pop_back();
  return out;
}




template<typename HilbertField>
void model_instance<HilbertField>::write_hdf5(H5::Group& grp)
{
  if(!gf_solved) Green_function_solve();

  h5_write_attr(grp, "GS_energy",  GS_energy);
  h5_write_attr(grp, "GS_sectors", GS_string());
  string fmt;
  if(GF_solver == GF_format_CF)                                   fmt = "cf";
  else if(GF_solver == GF_format_MCF ||
          GF_solver == GF_format_Q_to_MCF)                       fmt = "mcf";
  else                                                            fmt = "bl";
  h5_write_attr(grp, "GF_format",  fmt);
  h5_write_attr(grp, "mixing",     mixing);
  h5_write_attr(grp, "complex_HS", (int)complex_Hilbert);

  H5::Group pg = grp.createGroup("params");
  for(auto& kv : value) h5_write_attr(pg, kv.first, kv.second);

  h5_write_attr(grp, "nstates", (int)states.size());
  int idx = 0;
  for(auto& st : states){
    H5::Group sg = grp.createGroup("state_" + to_string(idx++));
    st->write_hdf5(sg);
  }
}


template<typename HilbertField>
void model_instance<HilbertField>::read_hdf5(H5::Group& grp)
{
  GS_energy = h5_read_attr_double(grp, "GS_energy");
  string fmt = h5_read_attr_string(grp, "GF_format");
  if(fmt == "bl")       GF_solver = GF_format_BL;
  else if(fmt == "mcf") GF_solver = GF_format_MCF;
  else                  GF_solver = GF_format_CF;
  mixing = h5_read_attr_int(grp, "mixing");
  n_mixed = 1;
  if(mixing & HS_mixing::anomalous) n_mixed *= 2;
  if(mixing & HS_mixing::spin_flip) n_mixed *= 2;

  // Check parameter values
  H5::Group pg = grp.openGroup("params");
  for(auto& kv : value){
    if(pg.attrExists(kv.first)){
      double stored = h5_read_attr_double(pg, kv.first);
      if(std::abs(stored - kv.second) > MIDDLE_VALUE)
        cout << "WARNING : The value of " << kv.first
             << " from the HDF5 solution (" << stored
             << ") differs from the expected value (" << kv.second << ")." << endl;
    }
  }

  // Read states
  clear_states();
  int nstates = h5_read_attr_int(grp, "nstates");
  for(int i = 0; i < nstates; ++i){
    H5::Group sg = grp.openGroup("state_" + to_string(i));
    states.insert(make_shared<state<HilbertField>>(sg, the_model->group, mixing));
  }

  is_correlated = true;
  M.set_size(dim_GF);
  if(mixing & HS_mixing::up_down) M_down.set_size(dim_GF);
  Green_function_average();
  Green_function_density();
  gf_read   = true;
  gf_solved = true;
}


template<typename HilbertField>
pair<vector<double>, vector<complex<double>>> model_instance<HilbertField>::qmatrix(bool spin_down)
{
  if(GF_solver != GF_format_BL) qcm_ED_throw("Green function format is not Lehmann! Cannot output the Q matrix.");
  if(!gf_solved) Green_function_solve();
  if(states.size() > 1){
    cout << "Lehmann representation from a mixed state : ";
    for(auto& s : states) cout << s->sec << '\t' << s->weight << endl;
    // qcm_ED_throw("The ground state is not a pure state! ("+to_string(states.size())+" states). Cannot output the Q matrix.");
    Q_matrix<HilbertField> QQ;
    for(auto& s : states){
      shared_ptr<Green_function_set> gf;
      if(spin_down) gf = s->gf_down;
      else gf = s->gf;
      auto Q = dynamic_pointer_cast<Q_matrix_set<HilbertField>>(gf)->consolidated_qmatrix();
      Q.v.v *= sqrt(s->weight);
      QQ.append(Q);
    }
    return {QQ.e, to_complex(QQ.v.v)};
  }
  else{
    shared_ptr<Green_function_set> gf;
    if(spin_down) gf = (*states.begin())->gf_down;
    else gf = (*states.begin())->gf;
    auto Q = dynamic_pointer_cast<Q_matrix_set<HilbertField>>(gf)->consolidated_qmatrix();
    return {Q.e, to_complex(Q.v.v)};
  }
}


template<typename HilbertField>
pair<vector<double>, vector<complex<double>>> model_instance<HilbertField>::hybridization(bool spin_down)
{
  if(the_model->n_bath == 0) qcm_ED_throw("System has no bath. Hybridization not defined.");
  if(!hopping_solved) set_hopping_matrix(spin_down);
  vector<double> E(tb.r);
  if(spin_down){
    for(size_t i=0; i<tb.r; i++) E[i] = real<double>(tb_down(i,i));
    return {E, to_complex(tcb_down.v)};
  }
  for(size_t i=0; i<tb.r; i++) E[i] = real<double>(tb(i,i));
  return {E, to_complex(tcb.v)};
}
#endif /* model_instance_hpp */




template<typename HilbertField>
void model_instance<HilbertField>::print_wavefunction(ostream& fout)
{
  for (auto& s : states){
    if(the_model->is_factorized){
      auto B = the_model->provide_factorized_basis(s->sec);
      s->write_wavefunction(fout, *B);
    }
    else{
      auto B = the_model->provide_basis(s->sec);
      s->write_wavefunction(fout, *B);
    }
  }
}




template<typename HilbertField>
matrix<Complex> model_instance<HilbertField>::hopping_matrix_full(bool spin_down, bool diag)
{
  matrix<Complex> H(tc.r+tb.r);
  if(diag){
    if(spin_down){
      tc_down.move_sub_matrix(tc.r, tc.c, 0, 0, 0, 0, H);
      tb_down.move_sub_matrix(tb.r, tb.c, 0, 0, tc.r, tc.c, H);
      tcb_down.move_sub_matrix(tcb.r, tcb.c, 0, 0, 0, tc.c, H);
      tcb_down.move_sub_matrix_HC(tcb.r, tcb.c, 0, 0, tc.r, 0, H);
    }
    else{
      tc.move_sub_matrix(tc.r, tc.c, 0, 0, 0, 0, H);
      tb.move_sub_matrix(tb.r, tb.c, 0, 0, tc.r, tc.c, H);
      tcb.move_sub_matrix(tcb.r, tcb.c, 0, 0, 0, tc.c, H);
      tcb.move_sub_matrix_HC(tcb.r, tcb.c, 0, 0, tc.r, 0, H);
    }
  }
  else{
    if(spin_down){
      tc_down.move_sub_matrix(tc.r, tc.c, 0, 0, 0, 0, H);
      tb_nd_down.move_sub_matrix(tb.r, tb.c, 0, 0, tc.r, tc.c, H);
      tcb_nd_down.move_sub_matrix(tcb.r, tcb.c, 0, 0, 0, tc.c, H);
      tcb_nd_down.move_sub_matrix_HC(tcb.r, tcb.c, 0, 0, tc.r, 0, H);
    }
    else{
      tc.move_sub_matrix(tc.r, tc.c, 0, 0, 0, 0, H);
      tb_nd.move_sub_matrix(tb.r, tb.c, 0, 0, tc.r, tc.c, H);
      tcb_nd.move_sub_matrix(tcb.r, tcb.c, 0, 0, 0, tc.c, H);
      tcb_nd.move_sub_matrix_HC(tcb.r, tcb.c, 0, 0, tc.r, 0, H);
    }
  }
  return H;
}



template<typename HilbertField>
vector<tuple<int,int,double>> model_instance<HilbertField>::interactions()
{
  vector<tuple<int,int,double>> V;
  V.reserve(the_model->n_sites);
  for(auto& x : value){
    if(the_model->term.at(x.first)->is_interaction){
      auto elem = the_model->term.at(x.first)->matrix_elements();
      for(auto& y : elem){
        V.push_back(tuple<int,int,double>(y.r, y.c, y.v.real()*x.second));
      }
    }
  }
  return V;
}




template<typename HilbertField>
pair<matrix<Complex>, vector<uint64_t>> model_instance<HilbertField>::density_matrix_mixed(vector<int> sites)
{
  if(the_model->group->g > 1) qcm_ED_throw("computing the density matrix requires symmetries to be turned off!");

  if(!gs_solved) low_energy_states();
  state<HilbertField> psi = **states.begin();
  shared_ptr<ED_mixed_basis> base = the_model->basis[psi.sec];

  // psi.write_wavefunction(cout, *base); // ***

  int nA = sites.size();
  int nB = the_model->n_orb - sites.size();
  vector<int> sitesB(0);
  for(int i=0; i<the_model->n_orb; i++){
    int j;
    for(j=0; j<sites.size(); j++) if(sites[j] == i) break;
    if(j == sites.size()) sitesB.push_back(i);
  }

  // constructing the masks for subsystems A and B
  uint64_t mask_A = 0L;
  uint64_t mask_B = 0L;

  for(int i=0; i<sites.size(); i++)
    mask_A += (1 << sites[i]);
  mask_A = mask_A + (mask_A << 32);
  vector<int> S(sites); S += 1;
  S = sitesB; S+=1;

  // First step: decomposing the basis into the two subsystems
  map<uint64_t, int> index_A;
  map<uint64_t, int> index_B;
  vector<uint64_t> binA;
  vector<uint64_t> binB;
  vector<vector<int>> listA;
  int i=0;
  int j=0;
  for(int k=0; k<base->dim; k++){
    uint64_t A = base->binlist[k].b & mask_A;
    uint64_t B = base->binlist[k].b ^ A;
    if (index_A.find(A) == index_A.end()){
      index_A[A] = i++;
      // binA.push_back(collapse(A,sites));
      binA.push_back(A);
    }
    if (index_B.find(B) ==  index_B.end()){
      index_B[B] = j++;
      listA.push_back(vector<int>(0));
      // binB.push_back(collapse(B,sitesB));
      binB.push_back(B);
    }
    listA[index_B[B]].push_back(index_A[A]);
  }
  int dim_A = i;
  int dim_B = j;

  matrix<Complex> rho(dim_A);
  // loop over states
  for(int j=0; j<dim_B; j++){
    vector<int> &v = listA[j];
    // cout << j << " : " << v << endl;
    for(int i=0; i<v.size(); i++){
      auto bI = binary_state(binA[v[i]] | binB[j]);
      int I = base->index(bI);
      for(int ip=0; ip<v.size(); ip++){
        auto bIp = binary_state(binA[v[ip]] | binB[j]);
        int Ip = base->index(bIp);
        rho(v[i], v[ip]) += conj(psi.psi[I])*psi.psi[Ip];
      }
    }
  }

  vector<uint64_t> binA_collapsed(dim_A);
  for(int i=0; i<dim_A; i++) binA_collapsed[i] = collapse(binA[i], sites);

  return {rho, binA_collapsed};
}




template<typename HilbertField>
pair<matrix<Complex>, vector<uint64_t>> model_instance<HilbertField>::density_matrix_factorized(vector<int> sites)
{
    // loop over states
    return {matrix<Complex>(0), vector<uint64_t>(0)};
}


template<typename HilbertField>
void model_instance<HilbertField>::merge_states()
{
  sector current_sector;
  auto merged_state = shared_ptr<state<HilbertField>>(new state<HilbertField>());

  vector<multimap<double, vector<HilbertField>>> M(the_model->group->g); // for each irrep, a multimap that stands for the global Q matrix
  cout << "merging " << states.size() << " states into one" << endl;
  double E = 0.0;
  for(auto &x : states){
    E += x->weight*x->energy;
    shared_ptr<Q_matrix_set<HilbertField>> Q = dynamic_pointer_cast<Q_matrix_set<HilbertField>>(x->gf);
    Q->merge(M);
  }
  merged_state->energy = E;
  vector<vector<double>> w(M.size());
  vector<matrix<HilbertField>> q(M.size());

  for(int i=0; i<M.size(); i++){
    size_t map_size = M[i].size();
    w[i].resize(map_size);
    q[i].set_size(n_mixed*the_model->group->site_irrep_dim[i], map_size);
    size_t j = 0;
    for(auto& x: M[i]){
      w[i][j] = x.first;
      for(int k=0; k<x.second.size(); k++) q[i](k,j) = x.second[k];
      j++;
    }
  }
  auto Qmerged = shared_ptr<Q_matrix_set<HilbertField>>(new Q_matrix_set<HilbertField>(the_model->group, mixing, w, q));

  Qmerged->streamline(true);
  merged_state->gf = Qmerged;

  if(mixing&HS_mixing::up_down){
    vector<multimap<double, vector<HilbertField>>> M_down(the_model->group->g);
    for(auto &x : states){
      shared_ptr<Q_matrix_set<HilbertField>> Q = dynamic_pointer_cast<Q_matrix_set<HilbertField>>(x->gf_down);
      Q->merge(M_down);
    }
    vector<vector<double>> w(M_down.size());
    vector<matrix<HilbertField>> q(M_down.size());

    for(int i=0; i<M_down.size(); i++){
      size_t map_size = M_down[i].size();
      w[i].resize(map_size);
      q[i].set_size(n_mixed*the_model->group->site_irrep_dim[i], map_size);
      size_t j = 0;
      for(auto& x: M_down[i]){
        w[i][j] = x.first;
        for(int k=0; k<x.second.size(); k++) q[i](k,j) = x.second[k];
        j++;
      }
    }
    auto Qmerged_down = shared_ptr<Q_matrix_set<HilbertField>>(new Q_matrix_set<HilbertField>(the_model->group, mixing, w, q));
    Qmerged_down->streamline(true);
    merged_state->gf_down = Qmerged_down;
  }

  states.clear();
  states.insert(merged_state);
}