37template<
typename T,
typename U>
concept ArmaContainer = std::is_floating_point_v<U> && (std::is_convertible_v<T, Mat<U>> || std::is_convertible_v<T, SpMat<U>>) ;
44 shared_ptr<MetaMat<T>> mat_a, mat_b;
47 explicit op_add(
const shared_ptr<MetaMat<T>>&
A)
51 op_add(
const shared_ptr<MetaMat<T>>&
A,
const shared_ptr<MetaMat<T>>& B)
69 , bracket(std::forward<
op_add<
T>>(B)) {}
90 template<std::invocable<fmat&> F>
int mixed_trs(mat& X, mat&& B, F trs) {
93 X = arma::zeros(size(B));
95 auto multiplier = norm(B);
101 auto residual = conv_to<fmat>::from(B / multiplier);
103 if(0 != (INFO = trs(residual)))
break;
105 const mat incre = multiplier * conv_to<mat>::from(residual);
109 suanpan_debug(
"Mixed precision algorithm multiplier: {:.5E}.\n", multiplier = arma::norm(B -= this->
operator*(incre)));
122 MetaMat(
const uword in_rows,
const uword in_cols,
const uword in_elem)
147 this->
at(K,
K) =
T(1);
152 [[nodiscard]]
virtual T max()
const = 0;
153 [[nodiscard]]
virtual Col<T>
diag()
const = 0;
164 virtual T&
unsafe_at(
const uword I,
const uword J) {
return this->
at(I, J); }
170 virtual T&
at(uword, uword) = 0;
172 [[nodiscard]]
virtual const T*
memptr()
const = 0;
183 const auto& bracket =
M.bracket;
184 if(
nullptr != bracket.mat_a) this->
scale_accu(M.scalar, bracket.mat_a);
185 if(
nullptr != bracket.mat_b) this->
scale_accu(M.scalar, bracket.mat_b);
189 const auto& bracket =
M.bracket;
190 if(
nullptr != bracket.mat_a) this->
scale_accu(-M.scalar, bracket.mat_a);
191 if(
nullptr != bracket.mat_b) this->
scale_accu(-M.scalar, bracket.mat_b);
204 template<ArmaContainer<T> C> Mat<T>
solve(C&& B) {
207 if(
SUANPAN_SUCCESS != this->
solve(X, std::forward<C>(B)))
throw std::runtime_error(
"fail to solve the system");
215 if(!
to_mat(*this).save(name, raw_ascii))
227 this->csc_condense();
229 X.zeros(arma::size(B));
231 unique_ptr<Preconditioner<T>> preconditioner;
232 if(
PreconditionerType::JACOBI == this->setting.preconditioner_type) preconditioner = std::make_unique<Jacobi<T>>(this->diag());
233#ifndef SUANPAN_SUPERLUMT
235 if(this->triplet_mat.is_empty()) preconditioner = std::make_unique<ILU<T>>(to_triplet_form<T, int>(
this));
236 else preconditioner = std::make_unique<ILU<T>>(this->triplet_mat);
239 else if(
PreconditionerType::NONE == this->setting.preconditioner_type) preconditioner = std::make_unique<UnityPreconditioner<T>>();
243 this->setting.preconditioner = preconditioner.get();
245 std::atomic_int code = 0;
249 Col<T> sub_x(X.colptr(I), X.n_rows, false, true);
250 const Col<T> sub_b(B.colptr(I), B.n_rows);
251 auto col_setting = setting;
252 code += GMRES(this, sub_x, sub_b, col_setting);
256 Col<T> sub_x(X.colptr(I), X.n_rows, false, true);
257 const Col<T> sub_b(B.colptr(I), B.n_rows);
258 auto col_setting = setting;
259 code += BiCGSTAB(this, sub_x, sub_b, col_setting);
261 else throw invalid_argument(
"no proper iterative solver assigned but somehow iterative solving is called");
266template<sp_d T> Mat<T>
to_mat(
const MetaMat<T>& in_mat) {
267 Mat<T> out_mat(in_mat.n_rows, in_mat.n_cols);
268 for(uword J = 0; J < in_mat.n_cols; ++J)
for(uword I = 0; I < in_mat.n_rows; ++I) out_mat(I, J) = in_mat(I, J);
272template<sp_d T> Mat<T>
to_mat(
const shared_ptr<MetaMat<T>>& in_mat) {
return to_mat(*in_mat); }
274template<sp_d data_t, sp_i index_t> Mat<data_t>
to_mat(
const triplet_form<data_t, index_t>& in_mat) {
275 Mat<data_t> out_mat(in_mat.n_rows, in_mat.n_cols, fill::zeros);
276 for(index_t I = 0; I < in_mat.n_elem; ++I) out_mat(in_mat.row(I), in_mat.col(I)) += in_mat.val(I);
280template<sp_d data_t, sp_i index_t> Mat<data_t>
to_mat(
const csr_form<data_t, index_t>& in_mat) {
281 Mat<data_t> out_mat(in_mat.n_rows, in_mat.n_cols, fill::zeros);
284 for(index_t I = 0; I < in_mat.n_elem; ++I) {
285 if(I >= in_mat.row_mem()[c_idx]) ++c_idx;
286 out_mat(c_idx - 1, in_mat.col_mem()[I]) += in_mat.val_mem()[I];
292template<sp_d data_t, sp_i index_t> Mat<data_t>
to_mat(
const csc_form<data_t, index_t>& in_mat) {
293 Mat<data_t> out_mat(in_mat.n_rows, in_mat.n_cols, fill::zeros);
296 for(index_t I = 0; I < in_mat.n_elem; ++I) {
297 if(I >= in_mat.col_mem()[c_idx]) ++c_idx;
298 out_mat(in_mat.row_mem()[I], c_idx - 1) += in_mat.val_mem()[I];
304template<sp_d data_t, sp_i index_t> triplet_form<data_t, index_t>
to_triplet_form(MetaMat<data_t>* in_mat) {
305 if(!in_mat->triplet_mat.is_empty())
return triplet_form<data_t, index_t>(in_mat->triplet_mat);
307 const sp_i auto n_rows = index_t(in_mat->n_rows);
308 const sp_i auto n_cols = index_t(in_mat->n_cols);
309 const sp_i auto n_elem = index_t(in_mat->n_elem);
311 triplet_form<data_t, index_t> out_mat(n_rows, n_cols, n_elem);
312 for(index_t J = 0; J < n_cols; ++J)
for(index_t I = 0; I < n_rows; ++I) out_mat.at(I, J) = in_mat->operator()(I, J);
317template<sp_d data_t, sp_i index_t> triplet_form<data_t, index_t>
to_triplet_form(
const shared_ptr<MetaMat<data_t>>& in_mat) {
return to_triplet_form<data_t, index_t>(in_mat.get()); }
Definition MetaMat.hpp:41
op_add(const shared_ptr< MetaMat< T > > &A, const shared_ptr< MetaMat< T > > &B)
Definition MetaMat.hpp:51
op_add(const shared_ptr< MetaMat< T > > &A)
Definition MetaMat.hpp:47
Definition MetaMat.hpp:56
op_scale(const T A, const shared_ptr< MetaMat< T > > &B)
Definition MetaMat.hpp:63
op_scale(const T A, op_add< T > &&B)
Definition MetaMat.hpp:67
Definition MetaMat.hpp:37
void for_each(const IT start, const IT end, F &&FN)
Definition utility.h:28
Definition SolverSetting.hpp:40
unsigned iterative_refinement
Definition SolverSetting.hpp:44
data_t tolerance
Definition SolverSetting.hpp:43
IterativeSolver iterative_solver
Definition SolverSetting.hpp:46
#define suanpan_debug(...)
Definition suanPan.h:307
constexpr auto SUANPAN_SUCCESS
Definition suanPan.h:172
constexpr auto SUANPAN_FAIL
Definition suanPan.h:173
#define suanpan_error(...)
Definition suanPan.h:309