Program Listing for File op.ipp

Return to documentation for file (SeQuant/domain/mbpt/op.ipp)

//
// Created by Eduard Valeyev on 2019-03-26.
//

#ifndef SEQUANT_DOMAIN_MBPT_OP_IPP
#define SEQUANT_DOMAIN_MBPT_OP_IPP

#include <SeQuant/core/tensor_canonicalizer.hpp>
#include <SeQuant/domain/mbpt/op.hpp>

namespace sequant {
namespace mbpt {

template <typename QuantumNumbers, Statistics S>
Operator<QuantumNumbers, S>::Operator() = default;

template <typename QuantumNumbers, Statistics S>
Operator<QuantumNumbers, S>::Operator(
    std::function<std::wstring_view()> label_generator,
    std::function<ExprPtr()> tensor_form_generator,
    std::function<void(QuantumNumbers&)> qn_action)
    : base_type(std::move(label_generator), std::move(tensor_form_generator)),
      qn_action_(std::move(qn_action)) {}

template <typename QuantumNumbers, Statistics S>
Operator<QuantumNumbers, S>::Operator(
    std::function<std::wstring_view()> label_generator,
    std::function<ExprPtr()> tensor_form_generator,
    std::function<void(QuantumNumbers&)> qn_action, size_t batch_idx_rank)
    : Operator(std::move(label_generator), std::move(tensor_form_generator),
               std::move(qn_action)) {
  mbpt::check_for_batching_space();
  SEQUANT_ASSERT(batch_idx_rank != 0 &&
                 "Operator: batch_idx_rank cannot be zero");
  // make aux ordinals [1 to batch_idx_rank]
  batch_ordinals_ = ranges::views::iota(1ul, 1ul + batch_idx_rank) |
                    ranges::to<container::svector<std::size_t>>();
}

template <typename QuantumNumbers, Statistics S>
Operator<QuantumNumbers, S>::Operator(
    std::function<std::wstring_view()> label_generator,
    std::function<ExprPtr()> tensor_form_generator,
    std::function<void(QuantumNumbers&)> qn_action,
    const container::svector<std::size_t>& batch_ordinals)
    : Operator(std::move(label_generator), std::move(tensor_form_generator),
               std::move(qn_action)) {
  mbpt::check_for_batching_space();
  SEQUANT_ASSERT(!batch_ordinals.empty() &&
                 "Operator: batch_ordinals cannot be empty");
  SEQUANT_ASSERT(ranges::is_sorted(batch_ordinals) &&
                 "Operator: batch_ordinals must be sorted");
  batch_ordinals_ = batch_ordinals;
}

template <typename QuantumNumbers, Statistics S>
Operator<QuantumNumbers, S>::~Operator() = default;

template <typename QuantumNumbers, Statistics S>
QuantumNumbers Operator<QuantumNumbers, S>::operator()(
    const QuantumNumbers& qns) const {
  QuantumNumbers result(qns);
  this->apply_to(result);
  return result;
}

template <typename QuantumNumbers, Statistics S>
QuantumNumbers& Operator<QuantumNumbers, S>::apply_to(
    QuantumNumbers& qns) const {
  SEQUANT_ASSERT(qn_action_);
  if (is_vacuum(qns)) {  // action on vacuum is trivial ...
    qn_action_(qns);
  } else {  // action on a {operator. product of operators} = use Wick's theorem
    QuantumNumbers qns_this;
    qn_action_(qns_this);
    qns = combine(qns_this, qns);
  }
  return qns;
}

template <typename QuantumNumbers, Statistics S>
bool Operator<QuantumNumbers, S>::static_less_than(const Expr& that) const {
  SEQUANT_ASSERT(that.is<this_type>());
  auto& that_op = that.as<this_type>();

  // compare cardinal tensor labels first, then QN ranks
  auto& cardinal_tensor_labels = TensorCanonicalizer::cardinal_tensor_labels();
  const auto this_label = this->label();
  const auto that_label = that_op.label();
  if (this_label == that_label) return this->less_than_rank_of(that_op);
  const auto this_is_cardinal_it = ranges::find_if(
      cardinal_tensor_labels,
      [&this_label](const std::wstring& l) { return l == this_label; });
  const auto this_is_cardinal =
      this_is_cardinal_it != ranges::end(cardinal_tensor_labels);
  const auto that_is_cardinal_it = ranges::find_if(
      cardinal_tensor_labels,
      [&that_label](const std::wstring& l) { return l == that_label; });
  const auto that_is_cardinal =
      that_is_cardinal_it != ranges::end(cardinal_tensor_labels);
  if (this_is_cardinal && that_is_cardinal) {
    if (this_is_cardinal_it != that_is_cardinal_it)
      return this_is_cardinal_it < that_is_cardinal_it;
    else
      return this->less_than_rank_of(that_op);
  } else if (this_is_cardinal && !that_is_cardinal)
    return true;
  else if (!this_is_cardinal && that_is_cardinal)
    return false;
  else {  // !this_is_cardinal && !that_is_cardinal
    if (this_label == that_label)
      return this->less_than_rank_of(that_op);
    else
      return this_label < that_label;
  }
}

template <typename QuantumNumbers, Statistics S>
bool Operator<QuantumNumbers, S>::commutes_with_atom(const Expr& that) const {
  SEQUANT_ASSERT(that.is_cnumber() || that.is<this_type>());
  if (that.is_cnumber())
    return true;
  else {
    auto& that_op = that.as<this_type>();

    // if this has annihilators/creators in same space as that has
    // creator/annihilators return false

    auto delta_this = (*this)();
    auto delta_that = (that_op)();

    SEQUANT_ASSERT(this->size() % 2 == 0 && that.size() == this->size());

    return combine(delta_this, delta_that) == combine(delta_that, delta_this);
  }
}

template <typename QuantumNumbers, Statistics S>
void Operator<QuantumNumbers, S>::adjoint() {
  const auto dN = (*this)(QuantumNumbers{});
  using qnc_t = std::decay_t<decltype(dN)>;
  static_assert(std::is_same_v<QuantumNumbers, qnc_t>,
                "mbpt::Operator::adjoint: QuantumNumbers type mismatch");

  // grab label and update according to adjoint flag
  auto lbl = std::wstring(this->label());
  if (lbl.back() == sequant::adjoint_label) {
    SEQUANT_ASSERT(is_adjoint_);
    lbl.pop_back();
  } else {
    SEQUANT_ASSERT(!is_adjoint_);
    lbl.push_back(sequant::adjoint_label);
  }

  const auto tnsr = this->tensor_form();
  *this =
      Operator{[=]() -> std::wstring_view { return lbl; },  // label_generator
               [=]() -> ExprPtr {
                 return sequant::adjoint(tnsr);  // tensor_form_generator
               },
               [=](qnc_t& qn) {
                 qn += sequant::adjoint(dN);
                 return qn;  // qn_action
               }};
  this->is_adjoint_ = !this->is_adjoint_;  // toggle adjoint flag
}

template <typename QuantumNumbers, Statistics S>
bool Operator<QuantumNumbers, S>::less_than_rank_of(
    const this_type& that) const {
  return (*this)(QuantumNumbers{}) < that(QuantumNumbers{});
}

template <typename QuantumNumbers, Statistics S>
Expr::type_id_type Operator<QuantumNumbers, S>::type_id() const {
  return Expr::get_type_id<this_type>();
};

template <typename QuantumNumbers, Statistics S>
ExprPtr Operator<QuantumNumbers, S>::clone() const {
  return ex<this_type>(*this);
}

// Expresses general operators in human interpretable form. for example:
// \hat{T}_2 is a particle conserving 2-body excitation operator a non-particle
// conserving operator \hat{R}_2_1 implies that two particles are created
// followed by a single hole creation. conversely \hat{R}_1_2 implies the that
// only one particle is annihilated followed by two holes being created. The
// rule being, that for non-particle conserving operators, the first position
// indicates where the quasiparticle is going to and the second position
// indicates where it comes from. for the case of adjoint operators, the adjoint
// is represented by the symbol ⁺ and superscripting the quasi-particle numbers.
// for example: hat{R⁺}^{1,2}} For operators in which one or more
// quasi-particles has only partial coverage in the particle_space or
// hole_space, this notation is unsuitable, and we default to level printing of
// the operator.
template <typename QuantumNumbers, Statistics S>
std::wstring Operator<QuantumNumbers, S>::to_latex() const {
  return sequant::to_latex(*this);
}

template <typename QuantumNumbers, Statistics S>
Expr::hash_type Operator<QuantumNumbers, S>::memoizing_hash() const {
  auto compute_hash = [this]() {
    using std::begin;
    using std::end;
    auto qns = (*this)(QuantumNumbers{});
    auto val = sequant::hash::value(qns);
    sequant::hash::combine(val, std::wstring(this->label()));
    if (batch_ordinals()) {
      const auto ordinals = batch_ordinals().value();
      sequant::hash::range(val, begin(ordinals), end(ordinals));
    }
    return val;
  };
  if (!this->hash_value_) {
    this->hash_value_ = compute_hash();
  } else {
    SEQUANT_ASSERT(*(this->hash_value_) == compute_hash());
  }
  return *(this->hash_value_);
}

template <typename QuantumNumbers, Statistics S>
bool Operator<QuantumNumbers, S>::static_equal(const Expr& other_expr) const {
  const auto& other =
      static_cast<const Operator<QuantumNumbers, S>&>(other_expr);
  return this->label() == other.label() &&
         (*this)(QuantumNumbers{}) == other(QuantumNumbers{}) &&
         this->batch_ordinals() == other.batch_ordinals();
}

}  // namespace mbpt
}  // namespace sequant

#endif  // SEQUANT_DOMAIN_MBPT_OP_IPP