https://mooseframework.inl.gov
MooseError.C
Go to the documentation of this file.
1 //* This file is part of the MOOSE framework
2 //* https://mooseframework.inl.gov
3 //*
4 //* All rights reserved, see COPYRIGHT for full restrictions
5 //* https://github.com/idaholab/moose/blob/master/COPYRIGHT
6 //*
7 //* Licensed under LGPL 2.1, please see LICENSE for details
8 //* https://www.gnu.org/licenses/lgpl-2.1.html
9 
10 #include "MooseError.h"
11 #include "MooseUtils.h"
12 #include "MooseVariable.h"
13 #include "Registry.h"
14 
15 #include "libmesh/string_to_enum.h"
16 
17 using namespace libMesh;
18 
19 namespace moose
20 {
21 
22 namespace internal
23 {
24 
25 std::string
27 {
28  std::stringstream ss;
29  ss << libMesh::Utility::enum_to_string<FEFamily>(var1.feType().family) << ",ORDER"
30  << var1.feType().order
31  << " != " << libMesh::Utility::enum_to_string<FEFamily>(var2.feType().family) << ",ORDER"
32  << var2.feType().order;
33  return ss.str();
34 }
35 
36 std::string
37 mooseMsgFmt(const std::string & msg, const std::string & title, const std::string & color)
38 {
39  std::ostringstream oss;
40  oss << "\n" << color << "\n" << title << "\n" << msg << COLOR_DEFAULT << "\n";
41  return oss.str();
42 }
43 
44 std::string
45 mooseMsgFmt(const std::string & msg, const std::string & color)
46 {
47  std::ostringstream oss;
48  oss << "\n" << color << "\n" << msg << COLOR_DEFAULT << "\n";
49  return oss.str();
50 }
51 
52 [[noreturn]] void
53 mooseErrorRaw(std::string msg,
54  const std::string & prefix /* = "" */,
55  const hit::Node * node /* = nullptr */,
56  const bool show_trace /* = true */)
57 {
59  throw MooseRuntimeError(msg, node);
60 
61  // Atomic that will be set as soon as any thread hits this method
62  static std::atomic_flag aborting = ATOMIC_FLAG_INIT;
63 
64  // Per-thread bool that will avoid us getting stuck in pause() if one
65  // thread hits this method twice.
66  thread_local static bool this_thread_aborting = false;
67 
68  // This branch will be hit after another thread has already set the atomic.
69  // MPI_Abort, despite its name, does not behave like std::abort but instead
70  // calls exit handlers and destroys statics. So we don't want to touch
71  // anything static at this point. We'll just wait until the winning thread
72  // has incurred program exit
73  if (aborting.test_and_set(std::memory_order_acq_rel) && !this_thread_aborting)
74  {
75  // Waiting for the other thread(s), not burning CPU
76  for (;;)
77  pause();
78  }
79  // We're the first thread to hit this method (we set the atomic), so we're
80  // responsible for dumping the error and trace(s) while the remaining
81  // threads wait for us to exit (via libmesh_abort())
82  else
83  {
84  // No, really, terminate. libmesh_abort() may throw an
85  // exception, and if we catch that exception (or a bad_alloc, or
86  // anything thrown) and mooseError because we didn't understand
87  // it, we just want to continue up the stack.
88  if (this_thread_aborting)
89  libmesh_abort();
90 
91  this_thread_aborting = true;
92 
93  // Output the message if there is one, but flush it without the trace
94  // as trace retrieval can be slow in some circumstances and we want to
95  // get the error message out ASAP
96  if (!msg.empty())
97  {
98  // If we have a node available, add in the hit context (file location)
99  if (node)
100  msg = Moose::hitMessagePrefix(*node) + msg;
101 
102  msg = mooseMsgFmt(msg, "*** ERROR ***", COLOR_RED) + "\n";
103  if (!prefix.empty()) // multiapp prefix
104  MooseUtils::indentMessage(prefix, msg);
105 
106  {
107  Threads::spin_mutex::scoped_lock lock(moose_stream_lock);
108  Moose::err << msg << std::flush;
109  }
110  }
111 
112  // Print the trace if enabled and on a single rank
113  if (show_trace && libMesh::global_n_processors() == 1)
114  {
115  std::ostringstream oss;
116  print_trace(oss);
117  auto trace = oss.str();
118  if (!prefix.empty()) // multiapp prefix
120 
121  {
122  Threads::spin_mutex::scoped_lock lock(moose_stream_lock);
123  Moose::err << trace << std::flush;
124  }
125  }
126 
127  // In parallel with libMesh configured with --enable-tracefiles, this will
128  // dump a trace for each rank to file
131 
132  libmesh_abort();
133  }
134 }
135 
136 void
137 mooseStreamAll(std::ostringstream &)
138 {
139 }
140 
141 std::string
142 formatMooseDocumentedError(const std::string & repo_name,
143  const unsigned int issue_num,
144  const std::string & msg)
145 {
146  const auto & repo_url = Registry::getRepositoryURL(repo_name);
147  std::stringstream oss;
148  oss << msg << "\n\nThis error is documented at " << repo_url << "/issues/" << issue_num << ".";
149  return oss.str();
150 }
151 
152 } // namespace internal
153 
154 void
155 translateMetaPhysicLError(const MetaPhysicL::LogicError &)
156 {
157  mooseError(
158  "We caught a MetaPhysicL error in while performing element or face loops. This is "
159  "potentially due to AD not having a sufficiently large derivative container size. To "
160  "increase the AD container size, you can run configure in the MOOSE root directory with the "
161  "'--with-derivative-size=<n>' option and then recompile. Other causes of MetaPhysicL logic "
162  "errors include evaluating functions where they are not defined or differentiable like sqrt "
163  "(which gets called for vector norm functions) or log with arguments <= 0");
164 }
165 
166 } // namespace moose
void mooseStreamAll(std::ostringstream &ss)
All of the following are not meant to be called directly - they are called by the normal macros (moos...
Definition: MooseError.C:137
const libMesh::FEType & feType() const
Get the type of finite element object.
processor_id_type global_n_processors()
std::string incompatVarMsg(MooseVariableFieldBase &var1, MooseVariableFieldBase &var2)
Builds and returns a string of the form:
Definition: MooseError.C:26
void mooseError(Args &&... args)
Emit an error message with the given stringified, concatenated args and terminate the application...
Definition: MooseError.h:311
void translateMetaPhysicLError(const MetaPhysicL::LogicError &)
emit a relatively clear error message when we catch a MetaPhysicL logic error
Definition: MooseError.C:155
Real trace(const RealTensor &A, const unsigned int &dim)
OrderWrapper order
This class provides an interface for common operations on field variables of both FE and FV types wit...
The following methods are specializations for using the libMesh::Parallel::packed_range_* routines fo...
void write_traceout()
void indentMessage(const std::string &prefix, std::string &message, const char *color, bool indent_first_line, const std::string &post_prefix)
Definition: MooseUtils.C:738
void libmesh_abort()
void mooseErrorRaw(std::string msg, const std::string &prefix="", const hit::Node *node=nullptr, const bool show_trace=true)
Main callback for emitting a moose error.
Definition: MooseError.C:53
void print_trace(std::ostream &out_stream=std::cerr)
static const std::string & getRepositoryURL(const std::string &repo_name)
Returns the repository URL associated with repo_name.
Definition: Registry.C:189
std::string hitMessagePrefix(const hit::Node &node)
Get the prefix to be associated with a hit node for a message.
Definition: Moose.C:831
Exception to be thrown whenever we have _throw_on_error set and a mooseError() is emitted...
Definition: MooseError.h:117
libMesh::Threads::spin_mutex moose_stream_lock
Definition: MooseError.h:140
bool _throw_on_error
Variable to turn on exceptions during mooseError(), should only be used within MOOSE unit tests or wh...
Definition: Moose.C:844
std::string mooseMsgFmt(const std::string &msg, const std::string &title, const std::string &color)
Format a message for output with a title.
Definition: MooseError.C:37
std::string formatMooseDocumentedError(const std::string &repo_name, const unsigned int issue_num, const std::string &msg)
Formats a documented error.
Definition: MooseError.C:142