1#ifndef LINEARALGEBRA_SRC_MOLPRO_LINALG_ITSOLV_PROPOSE_RSPACE_H
2#define LINEARALGEBRA_SRC_MOLPRO_LINALG_ITSOLV_PROPOSE_RSPACE_H
3#include <molpro/linalg/itsolv/IterativeSolver.h>
4#include <molpro/linalg/itsolv/helper.h>
5#include <molpro/linalg/itsolv/subspace/Dimensions.h>
6#include <molpro/linalg/itsolv/subspace/ISubspaceSolver.h>
7#include <molpro/linalg/itsolv/subspace/IXSpace.h>
8#include <molpro/linalg/itsolv/subspace/QSpace.h>
9#include <molpro/linalg/itsolv/subspace/gram_schmidt.h>
10#include <molpro/linalg/itsolv/subspace/util.h>
11#include <molpro/linalg/itsolv/util.h>
12#include <molpro/linalg/itsolv/wrap_util.h>
13#include <molpro/profiler/Profiler.h>
19 for (
auto& p : params) {
20 auto dot = handler.
dot(p, p);
21 dot = std::sqrt(std::abs(dot));
23 handler.
scal(1. / dot, p);
39template <
typename value_type>
41 const std::vector<int>& remove_qspace,
Logger& logger) {
43 const auto nQd = remove_qspace.size();
44 const auto nSol = solutions.
rows();
46 for (
size_t i = 0; i < nSol; ++i) {
47 for (
size_t j = 0; j < nQd; ++j) {
48 solutions_proj(i, j) = solutions(i, dims.
oQ + remove_qspace[j]);
50 for (
size_t j = 0; j < dims.
nD; ++j) {
51 solutions_proj(i, nQd + j) = solutions(i, dims.
oD + j);
54 logger.
msg(
"nSol, nQd, nD " + std::to_string(nSol) +
", " + std::to_string(nQd) +
", " + std::to_string(dims.
nD),
56 return solutions_proj;
73template <
typename value_type>
79 const auto nSol = solutions_proj.
rows();
80 const auto nQd = remove_qspace.size();
82 for (
size_t i = 0; i < nSol; ++i) {
83 for (
size_t ii = 0; ii <= i; ++ii) {
84 for (
size_t j = 0; j < nQd; ++j) {
85 for (
size_t k = 0; k < nQd; ++k) {
86 overlap_proj(i, ii) += solutions_proj(i, j) * solutions_proj(ii, k) *
87 overlap(dims.
oQ + remove_qspace[j], dims.
oQ + remove_qspace[k]);
89 for (
size_t k = 0; k < dims.
nD; ++k) {
90 overlap_proj(i, ii) +=
91 solutions_proj(i, j) * solutions_proj(ii, nQd + k) * overlap(dims.
oQ + remove_qspace[j], dims.
oD + k);
94 for (
size_t j = 0; j < dims.
nD; ++j) {
95 for (
size_t k = 0; k < dims.
nD; ++k) {
96 overlap_proj(i, ii) +=
97 solutions_proj(i, nQd + j) * solutions_proj(ii, nQd + k) * overlap(dims.
oD + j, dims.
oD + k);
99 for (
size_t k = 0; k < nQd; ++k) {
100 overlap_proj(i, ii) +=
101 solutions_proj(i, nQd + j) * solutions_proj(ii, k) * overlap(dims.
oD + j, dims.
oQ + remove_qspace[k]);
104 overlap_proj(ii, i) = overlap_proj(i, ii);
117template <
typename value_type,
typename value_type_abs>
119 const value_type_abs norm_thresh,
Logger& logger) {
121 const auto nSol = parameters.
rows();
122 auto norm_proj = std::vector<value_type_abs>(nSol, 0.);
123 for (
size_t i = 0; i < nSol; ++i)
124 norm_proj[i] = std::sqrt(std::abs(overlap(i, i)));
125 for (
size_t i = 0, j = 0; i < nSol; ++i) {
126 if (norm_proj[i] > norm_thresh) {
127 parameters.
row(j).scal(1. / norm_proj[i]);
128 overlap.col(j).scal(1. / norm_proj[i]);
129 overlap.row(j).scal(1. / norm_proj[i]);
133 overlap.remove_row_col(j, j);
134 std::stringstream ss;
135 ss << std::setprecision(3) <<
"remove projected solution parameter i = " << i <<
", norm = " << norm_proj[i];
140 logger.
msg(
"norm = ", std::begin(norm_proj), std::end(norm_proj),
Logger::Info);
141 logger.
msg(
"parameters after normalisation = " + as_string(parameters),
Logger::Info);
142 logger.
msg(
"overlap of parameters = " + as_string(overlap),
Logger::Info);
158template <
typename value_type,
typename value_type_abs>
164 value_type* m =
const_cast<std::vector<value_type>&
>(overlap_proj.
data()).data();
166 std::numeric_limits<value_type_abs>::max(),
true);
167 svd_vecs.remove_if([&svd_thresh](
const auto& el) {
return el.value < svd_thresh; });
168 svd_vecs.sort([](
const auto& lt,
const auto& rt) {
return lt.value < rt.value; });
169 const auto nD = svd_vecs.size();
170 const auto nX = solutions_proj.
cols();
172 auto svd = svd_vecs.begin();
173 for (
size_t i = 0; i < nD; ++i, ++svd)
174 for (
size_t j = 0; j < overlap_proj.
cols(); ++j)
175 for (
size_t k = 0; k < nX; ++k)
176 solutions_stable(i, k) += svd->v[j] * solutions_proj(j, k);
177 logger.
msg(
"nS without null space = " + std::to_string(solutions_stable.rows()),
Logger::Debug);
178 return solutions_stable;
190template <
typename value_type>
194 const auto nDnew = solutions_proj.
rows();
195 const auto nQd = remove_qspace.size();
196 const auto nQ = dims.
nQ - nQd;
198 for (
size_t i = 0; i < dims.
nD; ++i) {
199 ov.remove_row_col(dims.
oD, dims.
oD);
201 auto is_Qdelete = [&remove_qspace](
size_t i) {
202 return std::find(begin(remove_qspace), end(remove_qspace), i) != end(remove_qspace);
204 for (
size_t i = 0, j = 0; i < dims.
nQ; ++i) {
206 ov.remove_row_col(dims.
oQ + j, dims.
oQ + j);
210 const auto oDnew = dims.
nP + nQ + nR;
211 ov.resize({oDnew + nDnew, oDnew + nDnew});
218 auto accumulate_ov_offdiag = [&](
size_t i,
size_t j,
size_t jj) {
219 for (
size_t k = 0; k < nQd; ++k)
220 ov(oDnew + i, j) += solutions_proj(i, k) * overlap(jj, dims.
oQ + remove_qspace[k]);
221 for (
size_t k = 0; k < dims.
nD; ++k)
222 ov(oDnew + i, j) += solutions_proj(i, nQd + k) * overlap(jj, dims.
oD + k);
223 ov(j, oDnew + i) = ov(oDnew + i, j);
225 for (
size_t i = 0; i < nDnew; ++i) {
226 for (
size_t j = 0; j < dims.
nP; ++j)
227 accumulate_ov_offdiag(i, j, dims.
oP + j);
228 for (
size_t j = 0, jj = 0; j < dims.
nQ; ++j)
230 accumulate_ov_offdiag(i, dims.
nP + jj++, dims.
oQ + j);
231 for (
size_t j = 0; j < nR; ++j)
232 accumulate_ov_offdiag(i, dims.
nP + nQ + j, dims.
nX + j);
234 for (
size_t i = 0; i < nDnew; ++i) {
235 for (
size_t j = 0; j <= i; ++j) {
236 for (
size_t k = 0; k < nQd; ++k) {
237 for (
size_t l = 0; l < nQd; ++l)
238 ov(oDnew + i, oDnew + j) += solutions_proj(i, k) * solutions_proj(j, l) *
239 overlap(dims.
oQ + remove_qspace[k], dims.
oQ + remove_qspace[l]);
240 for (
size_t l = 0; l < dims.
nD; ++l)
241 ov(oDnew + i, oDnew + j) +=
242 solutions_proj(i, k) * solutions_proj(j, nQd + l) * overlap(dims.
oQ + remove_qspace[k], dims.
oD + l);
244 for (
size_t k = 0; k < dims.
nD; ++k) {
245 for (
size_t l = 0; l < nQd; ++l)
246 ov(oDnew + i, oDnew + j) +=
247 solutions_proj(i, nQd + k) * solutions_proj(j, l) * overlap(dims.
oD + k, dims.
oQ + remove_qspace[l]);
248 for (
size_t l = 0; l < dims.
nD; ++l)
249 ov(oDnew + i, oDnew + j) +=
250 solutions_proj(i, nQd + k) * solutions_proj(j, nQd + l) * overlap(dims.
oD + k, dims.
oD + l);
252 ov(oDnew + j, oDnew + i) = ov(oDnew + i, oDnew + j);
271template <
class R,
class Q,
class P,
typename value_type>
276 const auto nP = pparams.size(), nQ = qparams.size(), nD = dparams.size(), nN = params.size();
277 const auto nX = nP + nQ + nD + nN;
279 const auto oQ = oP + nP;
280 const auto oD = oQ + nQ;
281 const auto oN = oD + nD;
288 auto copy_upper_to_lower = [&ov, oN, nN](
size_t oX,
size_t nX) {
289 for (
size_t i = 0; i < nX; ++i)
290 for (
size_t j = 0; j < nN; ++j)
291 ov(oX + i, oN + j) = ov(oN + j, oX + i);
293 copy_upper_to_lower(oP, nP);
294 copy_upper_to_lower(oQ, nQ);
295 copy_upper_to_lower(oD, nD);
310template <
typename value_type>
314 using value_type_abs =
decltype(std::abs(solutions(0, 0)));
315 auto q_delete = std::vector<int>{};
316 auto q_indices = std::vector<int>(dims.
nQ);
317 std::iota(begin(q_indices), end(q_indices), 0);
318 const auto nSol = solutions.
rows();
319 while (q_indices.size() > max_size_qspace) {
320 auto max_contrib_to_solution = std::vector<value_type_abs>{};
321 for (
auto i : q_indices) {
322 auto contrib = std::vector<value_type_abs>(nSol);
323 for (
size_t j = 0; j < nSol; ++j)
324 contrib[j] = std::abs(solutions(j, dims.
oQ + i));
325 max_contrib_to_solution.emplace_back(*std::max_element(begin(contrib), end(contrib)));
327 auto it_min = std::min_element(begin(max_contrib_to_solution), end(max_contrib_to_solution));
328 auto i = std::distance(begin(max_contrib_to_solution), it_min);
329 q_delete.push_back(q_indices.at(i));
330 q_indices.erase(begin(q_indices) + i);
331 logger.
msg(
"contribution to solutions =", std::begin(max_contrib_to_solution), std::end(max_contrib_to_solution),
349template <
class R,
class Q,
class P,
typename value_type,
typename value_type_abs>
351 const std::vector<int>& q_delete,
const value_type_abs norm_thresh,
362 auto svd_vecs =
svd_system(overlap_full_subspace.rows(), overlap_full_subspace.cols(),
363 array::Span(&overlap_full_subspace(0, 0), overlap_full_subspace.size()), svd_thresh,
true);
364 assert(svd_vecs.empty() &&
"P+Q+D subspace should be stable by construction");
365 const auto nD = solutions_proj.rows();
366 const auto nQd = q_delete.size();
367 assert(nQd + dims.nD == solutions_proj.cols());
370 std::vector<Q> dparams_new, dactions_new;
373 Q
const* q =
nullptr;
374 if (!qparams.empty())
375 q = &qparams.front().get();
376 else if (!dparams.empty())
377 q = &dparams.front().get();
379 for (
size_t i = 0; i < nD; ++i) {
380 dparams_new.emplace_back(handler.
copy(*q));
381 dactions_new.emplace_back(handler.
copy(*q));
382 handler.
fill(0, dparams_new.back());
383 handler.
fill(0, dactions_new.back());
387 for (
size_t i = 0; i < nD; ++i) {
388 for (
size_t j = 0; j < q_delete.size(); ++j) {
389 handler.
axpy(solutions_proj(i, j), qparams.at(q_delete[j]), dparams_new.at(i));
390 handler.
axpy(solutions_proj(i, j), qactions.at(q_delete[j]), dactions_new.at(i));
392 for (
size_t j = 0; j < dims.nD; ++j) {
393 handler.
axpy(solutions_proj(i, nQd + j), dparams.at(j), dparams_new.at(i));
394 handler.
axpy(solutions_proj(i, nQd + j), dactions.at(j), dactions_new.at(i));
397 for (
size_t i = 0; i < nD; ++i) {
398 auto norm = std::sqrt(std::abs(handler.
dot(dparams_new.at(i), dparams_new.at(i))));
399 handler.
scal(1. / norm, dparams_new[i]);
400 handler.
scal(1. / norm, dactions_new[i]);
402 return std::make_tuple(std::move(dparams_new), std::move(dactions_new));
421template <
class R,
class Q,
class P,
typename value_type,
typename value_type_abs>
424 const CVecRef<Q>& dparams,
const value_type_abs norm_thresh,
427 const auto nR = rparams.size(), nP = pparams.size(), nQ = qparams.size(), nD = dparams.size();
429 assert(nP == dims.
nP && nQ == dims.
nQ && nD == dims.
nD);
430 auto orthogonalise = [&overlap, &rparams, nR](
const auto& xparams,
auto& handler,
const size_t oX,
const size_t nX) {
431 for (
size_t i = 0; i < nX; ++i) {
432 auto norm = std::abs(overlap(oX + i, oX + i));
434 auto dot_mat = handler.gemm_inner(
cwrap(rparams),
cwrap_arg(xparams.at(i).get()));
435 std::pair<size_t, size_t> mcoeff_dim = std::make_pair(1, nR);
437 for (
size_t j = 0; j < nR; ++j) {
438 mcoeff(0, j) = -mcoeff(0, j) / norm;
440 handler.gemm_outer(mcoeff,
cwrap_arg(xparams.at(i).get()), rparams);
445 prof->start(
"orthoganalise");
446 orthogonalise(pparams, handlers.
rp(), dims.
oP, nP);
447 orthogonalise(qparams, handlers.
rq(), dims.
oQ, nQ);
448 orthogonalise(dparams, handlers.
rq(), dims.
oD, nD);
450 prof->start(
"get null_params");
451 auto null_params = std::vector<int>{};
452 for (
size_t i = 0; i < nR; ++i) {
453 auto norm = std::sqrt(std::abs(handlers.
rr().dot(rparams[i], rparams[i])));
454 if (norm > norm_thresh) {
455 handlers.
rr().scal(1. / norm, rparams[i]);
456 for (
size_t j = i + 1; j < nR; ++j) {
457 auto ov = handlers.
rr().dot(rparams[i], rparams[j]);
458 handlers.
rr().axpy(-ov, rparams[i], rparams[j]);
461 null_params.push_back(i);
481template <
typename value_type,
typename value_type_abs>
483 const value_type_abs svd_thresh,
Logger& logger) {
485 prof->start(
"itsolv::svd_system");
487 auto redundant_params = std::vector<int>{};
488 auto rspace_indices = std::vector<int>(nR);
489 std::iota(std::begin(rspace_indices), std::end(rspace_indices), 0);
490 auto svd =
svd_system(overlap.rows(), overlap.cols(),
491 array::Span(
const_cast<value_type*
>(overlap.data().data()), overlap.size()), svd_thresh,
true);
493 prof->start(
"find redundant parameters");
494 for (
const auto& singular_system : svd) {
495 if (!rspace_indices.empty()) {
496 auto rspace_contribution = std::vector<value_type_abs>{};
497 for (
auto i : rspace_indices)
498 rspace_contribution.push_back(std::abs(singular_system.v.at(oR + i)));
499 auto it_min = std::max_element(std::begin(rspace_contribution), std::end(rspace_contribution));
500 auto imin = std::distance(std::begin(rspace_contribution), it_min);
501 redundant_params.push_back(rspace_indices[imin]);
502 rspace_indices.erase(std::begin(rspace_indices) + imin);
503 std::stringstream ss;
504 ss << std::setprecision(3) <<
"redundant parameter found, i = " << redundant_params.back()
505 <<
", svd.value = " << singular_system.value
506 <<
", svd.v[i] = " << singular_system.v[oR + redundant_params.back()];
511 return redundant_params;
517 auto new_indices =
find_ref(wparams, params);
518 auto new_working_set = std::vector<int>{};
519 for (
auto i : new_indices) {
520 new_working_set.emplace_back(working_set.at(i));
522 return new_working_set;
553template <
class R,
class Q,
class P,
typename value_type_abs>
561 logger.
msg(
"dimensions {nP, nQ, nD, nW} = " + std::to_string(xspace.
dimensions().
nP) +
", " +
565 profiler.
start(
"itsolv::ISubspaceSolver::solutions");
566 auto solutions = subspace_solver.
solutions();
569 logger.
msg(
"delete Q parameter indices = ", q_delete.begin(), q_delete.end(),
Logger::Debug);
570 if (!q_delete.empty()) {
571 auto prof = profiler.
push(
"construct_dspace");
572 auto [dparams, dactions] =
573 construct_dspace(solutions, xspace, q_delete, res_norm_thresh, svd_thresh, handlers.
qq(), logger);
574 std::sort(begin(q_delete), end(q_delete), std::greater<int>());
575 for (
auto iq : q_delete)
577 auto wdparams =
wrap(dparams);
578 auto wdactions =
wrap(dactions);
580 auto eigenvalues_ref = subspace_solver.
eigenvalues();
581 subspace_solver.
solve(xspace, solutions.rows());
582 auto eigval_error = std::vector<double>{};
583 std::transform(std::begin(eigenvalues_ref), std::end(eigenvalues_ref), std::begin(subspace_solver.
eigenvalues()),
584 std::back_inserter(eigval_error), [](
auto& e_ref,
auto& e_new) { return std::abs(e_ref - e_new); });
585 logger.
msg(
"eigenvalue error due to new D space = ", std::begin(eigval_error), std::end(eigval_error),
590 auto wresidual =
wrap(residuals.begin(), residuals.begin() + solver.
working_set().size());
591 profiler.
start(
"normalise");
594 profiler.
start(
"append_overlap_with_r");
595 prof->
start(
"append_overlap_with_r");
596 const auto full_overlap =
601 prof->
start(
"redundant_indices");
602 auto redundant_indices =
605 logger.
msg(
"redundant indices = ", std::begin(redundant_indices), std::end(redundant_indices),
Logger::Debug);
607 profiler.
start(
"modified_gram_schmidt");
608 prof->
start(
"modified_gram_schmidt");
609 auto null_param_indices =
615 logger.
msg(
"null parameters = ", std::begin(null_param_indices), std::end(null_param_indices),
Logger::Debug);
618 for (
size_t i = 0; i < wresidual.size(); ++i)
619 handlers.
rr().copy(parameters.at(i), wresidual.at(i));
620 profiler.
start(
"get_new_working_set");
623 return new_working_set;
Enhances various operations between pairs of arrays and allows dynamic code injection with uniform in...
Definition: ArrayHandler.h:162
virtual value_type dot(const AL &x, const AR &y)=0
virtual void scal(value_type alpha, AL &x)=0
virtual AL copy(const AR &source)=0
virtual void axpy(value_type alpha, const AR &x, AL &y)=0
virtual void fill(value_type alpha, AL &x)=0
Non-owning container taking a pointer to the data buffer and its size and exposing routines for itera...
Definition: Span.h:28
Class, containing a collection of array handlers used in IterativeSolver Provides a Builder sub-class...
Definition: ArrayHandlers.h:25
auto & qq()
Definition: ArrayHandlers.h:40
auto & rp()
Definition: ArrayHandlers.h:44
auto & rr()
Definition: ArrayHandlers.h:39
auto & rq()
Definition: ArrayHandlers.h:42
Base class defining the interface common to all iterative solvers.
Definition: IterativeSolver.h:200
virtual const std::vector< int > & working_set() const =0
Working set of roots that are not yet converged.
virtual CVecRef< Q > cparamsq() const =0
virtual CVecRef< P > cparamsp() const =0
virtual CVecRef< Q > cactionsd() const =0
SubspaceData data
Equation data in the subspace.
Definition: IXSpace.h:22
virtual void eraseq(size_t i)=0
Removes parameter i from Q subspace.
virtual CVecRef< Q > cactionsq() const =0
virtual CVecRef< Q > cparamsd() const =0
virtual const Dimensions & dimensions() const =0
virtual void update_dspace(VecRef< Q > ¶ms, VecRef< Q > &actions)=0
Updates D space with the new parameters.
Slice row(size_t i)
Access row slice.
Definition: Matrix.h:118
void remove_row(index_type row)
removes a row from the matrix
Definition: Matrix.h:142
const std::vector< T > & data() const &
Access the underlying data buffer.
Definition: Matrix.h:65
size_t size() const
Definition: Matrix.h:168
index_type rows() const
Definition: Matrix.h:165
index_type cols() const
Definition: Matrix.h:166
Profiler & start(const std::string &name)
Profiler & stop(const std::string &name="")
Proxy push(const std::string &name)
static std::shared_ptr< Profiler > single()
auto construct_projected_solutions_overlap(const subspace::Matrix< value_type > &solutions_proj, const subspace::Matrix< value_type > &overlap, const subspace::Dimensions &dims, const std::vector< int > &remove_qspace, Logger &logger)
Constructs overlap matrix for projected solutions.
Definition: propose_rspace.h:74
auto remove_null_projected_solutions(const subspace::Matrix< value_type > &solutions_proj, const subspace::Matrix< value_type > &overlap_proj, const value_type_abs svd_thresh, Logger &logger)
Transforms to a stable subspace of projected solutions via SVD.
Definition: propose_rspace.h:159
void remove_null_norm_and_normalise(subspace::Matrix< value_type > ¶meters, subspace::Matrix< value_type > &overlap, const value_type_abs norm_thresh, Logger &logger)
Removes parameters with norm less than threshold and normalises the rest.
Definition: propose_rspace.h:118
auto construct_full_subspace_overlap(const subspace::Matrix< value_type > &solutions_proj, const subspace::Dimensions &dims, const std::vector< int > &remove_qspace, const subspace::Matrix< value_type > &overlap, const size_t nR)
Constructs overlap matrix of P+Q+R+(projected solutions) subspaces, where Q is without removed parame...
Definition: propose_rspace.h:191
auto construct_projected_solution(const subspace::Matrix< value_type > &solutions, const subspace::Dimensions &dims, const std::vector< int > &remove_qspace, Logger &logger)
Projects solution from the full subspace on to Q_{delete} and current D space.
Definition: propose_rspace.h:40
Definition: IterativeSolverTemplate.h:19
auto limit_qspace_size(const subspace::Dimensions &dims, const size_t max_size_qspace, const subspace::Matrix< value_type > &solutions, Logger &logger)
Ensures that size Q space is within limit by proposing Q parameters for deletion.
Definition: propose_rspace.h:311
void normalise(const size_t n_roots, const VecRef< R > ¶ms, const VecRef< R > &actions, array::ArrayHandler< R, R > &handler, Logger &logger)
Definition: IterativeSolverTemplate.h:80
auto propose_rspace(IterativeSolver< R, Q, P > &solver, const VecRef< R > ¶meters, const VecRef< R > &residuals, subspace::IXSpace< R, Q, P > &xspace, subspace::ISubspaceSolver< R, Q, P > &subspace_solver, ArrayHandlers< R, Q, P > &handlers, Logger &logger, value_type_abs svd_thresh, value_type_abs res_norm_thresh, int max_size_qspace, molpro::profiler::Profiler &profiler)
Proposes new parameters for the subspace from the preconditioned residuals.
Definition: propose_rspace.h:554
auto get_new_working_set(const std::vector< int > &working_set, const CVecRef< R > ¶ms, const CVecRef< R > &wparams)
Returns new working set based on parameters included in wparams.
Definition: propose_rspace.h:516
auto append_overlap_with_r(const subspace::Matrix< value_type > &overlap, const CVecRef< R > ¶ms, const CVecRef< P > &pparams, const CVecRef< Q > &qparams, const CVecRef< Q > &dparams, ArrayHandlers< R, Q, P > &handlers, Logger &logger)
Constructs overlap of the full subspace by appending overlap with new parameters to the overlap of pr...
Definition: propose_rspace.h:272
auto construct_dspace(const subspace::Matrix< value_type > &solutions, const subspace::IXSpace< R, Q, P > &xspace, const std::vector< int > &q_delete, const value_type_abs norm_thresh, const value_type_abs svd_thresh, array::ArrayHandler< Q, Q > &handler, Logger &logger)
Constructs the new D space by projecting the solutions onto Qd+D subspace and ensuring they are well ...
Definition: propose_rspace.h:350
auto modified_gram_schmidt(const VecRef< R > &rparams, const subspace::Matrix< value_type > &overlap, const subspace::Dimensions &dims, const CVecRef< P > &pparams, const CVecRef< Q > &qparams, const CVecRef< Q > &dparams, const value_type_abs norm_thresh, ArrayHandlers< R, Q, P > &handlers, Logger &logger)
Orthogonalises R parameters against P+Q+D subspace (and themselves)
Definition: propose_rspace.h:422
auto redundant_parameters(const subspace::Matrix< value_type > &overlap, const size_t oR, const size_t nR, const value_type_abs svd_thresh, Logger &logger)
Deduces a set of parameters that are redundant due to linear dependencies.
Definition: propose_rspace.h:482
auto overlap(const CVecRef< R > &left, const CVecRef< Q > &right, array::ArrayHandler< Z, W > &handler) -> std::enable_if_t< detail::Z_and_W_are_one_of_R_and_Q< R, Q, Z, W >, Matrix< double > >
Calculates overlap matrix between left and right vectors.
Definition: util.h:49
void delete_parameters(std::vector< int > indices, Container ¶ms)
Removes parameters.
Definition: util.h:98
auto cwrap(ForwardIt begin, ForwardIt end)
Takes a begin and end iterators and returns a vector of references to each element.
Definition: wrap.h:52
auto cwrap_arg(T &&arg, S &&... args) -> std::enable_if_t< std::conjunction_v< std::is_same< decay_t< T >, decay_t< S > >... >, CVecRef< decay_t< T > > >
Constructs a vector of const reference wrappers with provided arguments.
Definition: wrap.h:149
auto wrap(ForwardIt begin, ForwardIt end)
Takes a begin and end iterators and returns a vector of references to each element.
Definition: wrap.h:32
std::list< SVD< value_type > > svd_system(size_t nrows, size_t ncols, const array::Span< value_type > &m, double threshold, bool hermitian=false, bool reduce_to_rank=false)
Performs singular value decomposition and returns SVD objects for singular values less than threshold...
Definition: helper-implementation.h:264
std::vector< std::reference_wrapper< const A > > CVecRef
Definition: wrap.h:14
std::vector< std::reference_wrapper< A > > VecRef
Definition: wrap.h:11
std::vector< size_t > find_ref(const VecRef< R > &wparams, ForwardIt begin, ForwardIt end)
Given wrapped references in wparams and a range of original parameters [begin, end),...
Definition: wrap_util.h:17
A dummy structured logger.
Definition: Logger.h:40
void msg(const std::string &message, Level log_lvl)
Definition: Logger.cpp:16
bool data_dump
highest level of warning/error that can be logged
Definition: Logger.h:69
static std::string scientific(double val)
Converts double to a string in scientific notation.
Definition: Logger.cpp:33
@ Info
Definition: Logger.h:49
@ Trace
Definition: Logger.h:49
@ Debug
Definition: Logger.h:49
@ Warn
Definition: Logger.h:49
Stores partitioning of XSpace into P, Q and R blocks with sizes and offsets for each one.
Definition: Dimensions.h:5
size_t oQ
Definition: Dimensions.h:13
size_t nD
Definition: Dimensions.h:10
size_t nX
Definition: Dimensions.h:11
size_t nQ
Definition: Dimensions.h:9
size_t nP
Definition: Dimensions.h:8
size_t oP
Definition: Dimensions.h:12
size_t oD
Definition: Dimensions.h:14
Manages solution of the subspace problem and storage of those solutions.
Definition: ISubspaceSolver.h:15
virtual void solve(IXSpace< R, Q, P > &xspace, size_t nroots_max)=0
Solve the subspace problem.
virtual const std::vector< value_type > & eigenvalues() const =0
Access eigenvalues from the last solve() call.
virtual const Matrix< value_type > & solutions() const =0
Access solutions from the last solve() call.