OGS
CreateTimeLoop.cpp
Go to the documentation of this file.
1// SPDX-FileCopyrightText: Copyright (c) OpenGeoSys Community (opengeosys.org)
2// SPDX-License-Identifier: BSD-3-Clause
3
4#include "CreateTimeLoop.h"
5
6#include <algorithm>
7#include <range/v3/algorithm/any_of.hpp>
8#include <range/v3/view/join.hpp>
9#include <string>
10
11#include "BaseLib/ConfigTree.h"
18#include "TimeLoop.h"
19
20namespace ProcessLib
21{
22std::unique_ptr<TimeLoop> createTimeLoop(
23 BaseLib::ConfigTree const& config, std::string const& output_directory,
24 const std::vector<std::unique_ptr<Process>>& processes,
25 const std::map<std::string, std::unique_ptr<NumLib::NonlinearSolverBase>>&
26 nonlinear_solvers,
27 std::vector<std::unique_ptr<MeshLib::Mesh>>& meshes,
28 bool const compensate_non_equilibrium_initial_residuum)
29{
31 auto output_config_tree = config.getConfigSubtreeOptional("output");
32 if (!output_config_tree)
33 {
34 INFO("No output section found.");
35 }
36 auto outputs =
37 output_config_tree
38 ? createOutput(*output_config_tree, output_directory, meshes)
40 : createOutputs(config.getConfigSubtree("outputs"),
41 output_directory, meshes);
42 auto const fixed_times_for_output =
44
45 if (auto const submesh_residuum_output_config_tree =
47 config.getConfigSubtreeOptional("submesh_residuum_output");
48 submesh_residuum_output_config_tree)
49 {
51 *submesh_residuum_output_config_tree, output_directory, meshes);
52
53 for (auto& process : processes)
54 {
55 auto const& residuum_vector_names =
56 process->initializeAssemblyOnSubmeshes(smroc.meshes);
57
58 for (auto const& name : residuum_vector_names | ranges::views::join)
59 {
60 smroc.output.doNotProjectFromBulkMeshToSubmeshes(
62 }
63 }
64
65 outputs.push_back(std::move(smroc.output));
66 }
67 else
68 {
69 // Submesh assembly must always be initialized.
70 for (auto& process : processes)
71 {
72 process->initializeAssemblyOnSubmeshes({});
73 }
74 }
75
76 auto per_process_data = createPerProcessData(
78 config.getConfigSubtree("processes"), processes, nonlinear_solvers,
79 compensate_non_equilibrium_initial_residuum, fixed_times_for_output);
80
81 const bool use_staggered_scheme =
82 ranges::any_of(processes.begin(), processes.end(),
83 [](auto const& process)
84 { return !(process->isMonolithicSchemeUsed()); });
85
86 std::unique_ptr<NumLib::StaggeredCoupling> staggered_coupling = nullptr;
87 if (use_staggered_scheme)
88 {
90 config, per_process_data);
91 }
92 else
93 {
94 if (per_process_data.size() > 1)
95 {
97 "The monolithic scheme is used. However more than one "
98 "process data tags (by name \"process\") inside tag "
99 "\"time_loop\" are defined for the staggered scheme. If you "
100 "want to use staggered scheme, please set the element of tag "
101 "\"<coupling_scheme>\" to \"staggered\".");
102 }
103 }
104
105 const auto minmax_iter =
106 std::minmax_element(per_process_data.begin(),
107 per_process_data.end(),
108 [](std::unique_ptr<ProcessData> const& a,
109 std::unique_ptr<ProcessData> const& b) {
110 return (a->timestep_algorithm->end() <
111 b->timestep_algorithm->end());
112 });
113 auto const start_time =
114 per_process_data[minmax_iter.first - per_process_data.begin()]
115 ->timestep_algorithm->begin();
116 auto const end_time =
117 per_process_data[minmax_iter.second - per_process_data.begin()]
118 ->timestep_algorithm->end();
119
120 return std::make_unique<TimeLoop>(
121 std::move(outputs), std::move(per_process_data),
122 std::move(staggered_coupling), start_time, end_time);
123}
124} // namespace ProcessLib
#define OGS_FATAL(...)
Definition Error.h:19
void INFO(fmt::format_string< Args... > fmt, Args &&... args)
Definition Logging.h:28
std::optional< ConfigTree > getConfigSubtreeOptional(std::string const &root) const
ConfigTree getConfigSubtree(std::string const &root) const
std::unique_ptr< StaggeredCoupling > createStaggeredCoupling(BaseLib::ConfigTree const &config, std::vector< std::unique_ptr< ProcessData > > const &per_process_data)
Create a StaggeredCoupling instance from the given configuration.
SubmeshResiduumOutputConfig createSubmeshResiduumOutputConfig(BaseLib::ConfigTree const &config, std::string const &output_directory, std::vector< std::unique_ptr< MeshLib::Mesh > > &meshes)
std::vector< std::unique_ptr< ProcessData > > createPerProcessData(BaseLib::ConfigTree const &config, std::vector< std::unique_ptr< Process > > const &processes, std::map< std::string, std::unique_ptr< NumLib::NonlinearSolverBase > > const &nonlinear_solvers, bool const compensate_non_equilibrium_initial_residuum, std::vector< double > const &fixed_times_for_output)
Output createOutput(OutputConfig &&oc, std::string const &output_directory, std::vector< std::unique_ptr< MeshLib::Mesh > > const &meshes)
std::unique_ptr< TimeLoop > createTimeLoop(BaseLib::ConfigTree const &config, std::string const &output_directory, const std::vector< std::unique_ptr< Process > > &processes, const std::map< std::string, std::unique_ptr< NumLib::NonlinearSolverBase > > &nonlinear_solvers, std::vector< std::unique_ptr< MeshLib::Mesh > > &meshes, bool const compensate_non_equilibrium_initial_residuum)
Builds a TimeLoop from the given configuration.
std::vector< double > calculateUniqueFixedTimesForAllOutputs(std::vector< Output > const &outputs)
std::vector< Output > createOutputs(const BaseLib::ConfigTree &output_configs, std::string const &output_directory, std::vector< std::unique_ptr< MeshLib::Mesh > > &meshes)