13#include <spdlog/fmt/bundled/ostream.h>
16#include <unordered_set>
27 return filenamePrefix.empty() || filenamePrefix.ends_with(
'/') ||
28 filenamePrefix.ends_with(
'\\')
34static void outputGlobalMatrix(
GlobalMatrix const& mat, std::ostream& os)
41static void outputGlobalVector(
GlobalVector const& vec, std::ostream& os)
43 os <<
"(" << vec.
size() <<
")\n";
47std::ofstream openGlobalMatrixOutputFile(std::string
const& filenamePrefix,
48 std::size_t
const counter,
49 double const t,
int const process_id,
50 std::string
const& which_matrix,
51 std::string
const& extension)
53 auto const filename = fmt::format(
54 "{}{}ogs_global_matrix_cnt_{:03}_t_{:g}_pcs_{}_{}.{}", filenamePrefix,
56 which_matrix, extension);
58 std::ofstream fh{filename};
62 OGS_FATAL(
"Could not open file `{}' for global matrix debug output",
66 fh << std::setprecision(std::numeric_limits<double>::max_digits10);
72 std::string
const& env_var)
74 char const*
const prefix = std::getenv(env_var.c_str());
75 return prefix ? std::make_optional(prefix) : std::nullopt;
80 return fmt::format(
"{}{}ogs_local_matrix.log", filenamePrefix,
84Eigen::Map<
const Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic,
89 static_cast<Eigen::Index
>(std::round(std::sqrt(entries.size())));
91 return {entries.data(), num_r_c, num_r_c};
100 std::string
const& warn_msg)
102 std::istringstream sstr{str};
103 std::unordered_set<std::size_t> result;
104 std::ptrdiff_t value;
106 while (sstr >> value)
108 [[likely]]
if (value >= 0)
110 result.insert(value);
114 if (!warn_msg.empty())
116 WARN(
"{}", warn_msg);
123 if (!sstr.eof() && !warn_msg.empty())
126 WARN(
"{}", warn_msg);
133 std::string
const& element_ids_str)
135 if (element_ids_str.empty())
140 if (element_ids_str ==
"*")
142 return [](std::size_t) {
return true; };
147 "Error parsing list of element ids for local matrix debug "
148 "output. We'll try to proceed anyway, as best as we can.");
150 if (element_ids.empty())
155 return [element_ids = std::move(element_ids)](std::size_t element_id)
156 {
return element_ids.contains(element_id); };
162 auto opt_prefix = getEnvironmentVariable(
"OGS_GLOBAL_MAT_OUT_PREFIX");
164 if (opt_prefix.has_value())
172 "You requested global matrix output (OGS_GLOBAL_MAT_OUT_PREFIX is "
173 "set), which is not yet implemented for PETSc matrices.");
193 process_id,
"M",
"mat");
196 outputGlobalMatrix(M, fh);
201 process_id,
"K",
"mat");
204 outputGlobalMatrix(K, fh);
209 process_id,
"b",
"vec");
212 outputGlobalVector(b, fh);
238 process_id,
"b",
"vec");
241 outputGlobalVector(b, fh);
246 process_id,
"Jac",
"mat");
249 outputGlobalMatrix(Jac, fh);
262 auto const opt_prefix = getEnvironmentVariable(
"OGS_LOCAL_MAT_OUT_PREFIX");
263 auto const opt_elements =
264 getEnvironmentVariable(
"OGS_LOCAL_MAT_OUT_ELEMENTS");
271 "Environment variable OGS_LOCAL_MAT_OUT_ELEMENTS is set, but "
272 "OGS_LOCAL_MAT_OUT_PREFIX is not. Local matrix debug output "
273 "will be disabled.");
281 "Environment variable OGS_LOCAL_MAT_OUT_PREFIX is set, but "
282 "OGS_LOCAL_MAT_OUT_ELEMENTS is not. Local matrix debug output "
283 "will be disabled.");
293 "Environment variable OGS_LOCAL_MAT_OUT_ELEMENTS not set "
294 "properly. Local matrix debug output will be disabled.");
298 auto const outputFilename = localMatrixOutputFilename(*opt_prefix);
304 "File `{}' for local matrix debug output could not be "
309 DBUG(
"Successfully opened local matrix debug output file {}.",
314 std::size_t
const element_id,
315 std::vector<double>
const& local_M_data,
316 std::vector<double>
const& local_K_data,
317 std::vector<double>
const& local_b_data)
324 std::lock_guard lock_guard{
mutex_};
328 DBUG(
"Writing to local matrix debug output file...");
330 fmt::print(fh,
"## t = {:.{}g}, process id = {}, element id = {}\n\n", t,
331 std::numeric_limits<double>::max_digits10, process_id,
334 if (!local_M_data.empty())
337 fmt::print(fh,
"# M\n{}\n\n", toSquareMatrixRowMajor(local_M_data));
340 if (!local_K_data.empty())
343 fmt::print(fh,
"# K\n{}\n\n", toSquareMatrixRowMajor(local_K_data));
346 if (!local_b_data.empty())
354 std::size_t
const element_id,
355 std::vector<double>
const& local_b_data,
356 std::vector<double>
const& local_Jac_data)
363 std::lock_guard lock_guard{
mutex_};
367 DBUG(
"Writing to local matrix debug output file...");
369 fmt::print(fh,
"## t = {:.{}g}, process id = {}, element id = {}\n\n", t,
370 std::numeric_limits<double>::max_digits10, process_id,
373 if (!local_b_data.empty())
379 if (!local_Jac_data.empty())
382 fmt::print(fh,
"# Jac\n{}\n\n\n",
383 toSquareMatrixRowMajor(local_Jac_data));
void DBUG(fmt::format_string< Args... > fmt, Args &&... args)
void WARN(fmt::format_string< Args... > fmt, Args &&... args)
IndexType getNumberOfColumns() const
return the number of columns
void write(const std::string &filename) const
printout this matrix for debugging
IndexType getNumberOfRows() const
return the number of rows
Global vector based on Eigen vector.
IndexType size() const
return a vector length
RawVectorType & getRawVector()
return a raw Eigen vector object
Eigen::Map< const Vector > toVector(std::vector< double > const &data, Eigen::VectorXd::Index size)
Creates an Eigen mapped vector from the given data vector.
std::unordered_set< std::size_t > parseSetOfSizeT(std::string const &str, std::string const &warn_msg)
std::function< bool(std::size_t)> createLocalMatrixOutputElementPredicate(std::string const &element_ids_str)
std::string localMatrixOutputFilename(std::string const &filenamePrefix)
Eigen::Map< const Eigen::Matrix< double, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor > > toSquareMatrixRowMajor(std::vector< double > entries)
static std::optional< std::string > getEnvironmentVariable(std::string const &env_var)
std::string getSeparatorAfterFilenamePrefix(std::string const &filenamePrefix)
std::string filenamePrefix_
void operator()(double const t, int const process_id, GlobalMatrix const &M, GlobalMatrix const &K, GlobalVector const &b)
bool isOutputRequested(std::size_t const element_id) const
std::function< bool(std::size_t)> output_element_predicate_
void operator()(double const t, int const process_id, std::size_t const element_id, std::vector< double > const &local_M_data, std::vector< double > const &local_K_data, std::vector< double > const &local_b_data)
std::ofstream outputFile_