xref: /petsc/src/sys/error/err.c (revision 47d993e71cae6087374231ffa40636568dd6db27)
17d0a6c19SBarry Smith 
2e5c89e4eSSatish Balay /*
3e5c89e4eSSatish Balay       Code that allows one to set the error handlers
4*47d993e7Ssuyashtn       Portions of this code are under:
5*47d993e7Ssuyashtn       Copyright (c) 2022 Advanced Micro Devices, Inc. All rights reserved.
6e5c89e4eSSatish Balay */
7af0996ceSBarry Smith #include <petsc/private/petscimpl.h> /*I "petscsys.h" I*/
8665c2dedSJed Brown #include <petscviewer.h>
9e5c89e4eSSatish Balay 
10e5c89e4eSSatish Balay typedef struct _EH *EH;
11e5c89e4eSSatish Balay struct _EH {
12efca3c55SSatish Balay   PetscErrorCode (*handler)(MPI_Comm, int, const char *, const char *, PetscErrorCode, PetscErrorType, const char *, void *);
13e5c89e4eSSatish Balay   void *ctx;
14e5c89e4eSSatish Balay   EH    previous;
15e5c89e4eSSatish Balay };
16e5c89e4eSSatish Balay 
1702c9f0b5SLisandro Dalcin static EH eh = NULL;
18e5c89e4eSSatish Balay 
19e5c89e4eSSatish Balay /*@C
20e5c89e4eSSatish Balay    PetscEmacsClientErrorHandler - Error handler that uses the emacsclient program to
21a5b23f4aSJose E. Roman     load the file where the error occurred. Then calls the "previous" error handler.
22e5c89e4eSSatish Balay 
23e5c89e4eSSatish Balay    Not Collective
24e5c89e4eSSatish Balay 
25e5c89e4eSSatish Balay    Input Parameters:
26a5b23f4aSJose E. Roman +  comm - communicator over which error occurred
27e32f2f54SBarry Smith .  line - the line number of the error (indicated by __LINE__)
28e5c89e4eSSatish Balay .  file - the file in which the error was detected (indicated by __FILE__)
29e5c89e4eSSatish Balay .  mess - an error text string, usually just printed to the screen
30e5c89e4eSSatish Balay .  n - the generic error number
31e5c89e4eSSatish Balay .  p - specific error number
32e5c89e4eSSatish Balay -  ctx - error handler context
33e5c89e4eSSatish Balay 
34e5c89e4eSSatish Balay    Options Database Key:
3510699b91SBarry Smith .   -on_error_emacs <machinename> - will contact machinename to open the Emacs client there
36e5c89e4eSSatish Balay 
37e5c89e4eSSatish Balay    Level: developer
38e5c89e4eSSatish Balay 
39811af0c4SBarry Smith    Note:
40e5c89e4eSSatish Balay    You must put (server-start) in your .emacs file for the emacsclient software to work
41e5c89e4eSSatish Balay 
42cc12936aSPatrick Sanan    Developer Note:
43811af0c4SBarry Smith    Since this is an error handler it cannot call `PetscCall()`; thus we just return if an error is detected.
44811af0c4SBarry Smith    But some of the functions it calls do perform error checking that may not be appropriate in a error handler call.
453bf036e2SBarry Smith 
46db781477SPatrick Sanan .seealso: `PetscError()`, `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscAttachDebuggerErrorHandler()`,
47db781477SPatrick Sanan           `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscReturnErrorHandler()`
48e5c89e4eSSatish Balay  @*/
49d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscEmacsClientErrorHandler(MPI_Comm comm, int line, const char *fun, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, void *ctx)
50d71ae5a4SJacob Faibussowitsch {
51e5c89e4eSSatish Balay   PetscErrorCode ierr;
52e5c89e4eSSatish Balay   char           command[PETSC_MAX_PATH_LEN];
53e5c89e4eSSatish Balay   const char    *pdir;
54e5c89e4eSSatish Balay   FILE          *fp;
55e5c89e4eSSatish Balay 
569371c9d4SSatish Balay   ierr = PetscGetPetscDir(&pdir);
5711cc89d2SBarry Smith   if (ierr) return ierr;
58efca3c55SSatish Balay   sprintf(command, "cd %s; emacsclient --no-wait +%d %s\n", pdir, line, file);
59e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
609371c9d4SSatish Balay   ierr = PetscPOpen(MPI_COMM_WORLD, (char *)ctx, command, "r", &fp);
6111cc89d2SBarry Smith   if (ierr) return ierr;
629371c9d4SSatish Balay   ierr = PetscPClose(MPI_COMM_WORLD, fp);
6311cc89d2SBarry Smith   if (ierr) return ierr;
64e5c89e4eSSatish Balay #else
65e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP_SYS, "Cannot run external programs on this machine");
66e5c89e4eSSatish Balay #endif
679371c9d4SSatish Balay   ierr = PetscPopErrorHandler();
6811cc89d2SBarry Smith   if (ierr) return ierr; /* remove this handler from the stack of handlers */
693bf036e2SBarry Smith   if (!eh) {
709371c9d4SSatish Balay     ierr = PetscTraceBackErrorHandler(comm, line, fun, file, n, p, mess, NULL);
7111cc89d2SBarry Smith     if (ierr) return ierr;
723bf036e2SBarry Smith   } else {
739371c9d4SSatish Balay     ierr = (*eh->handler)(comm, line, fun, file, n, p, mess, eh->ctx);
7411cc89d2SBarry Smith     if (ierr) return ierr;
753bf036e2SBarry Smith   }
7611cc89d2SBarry Smith   return 0;
77e5c89e4eSSatish Balay }
78e5c89e4eSSatish Balay 
79e5c89e4eSSatish Balay /*@C
80e5c89e4eSSatish Balay    PetscPushErrorHandler - Sets a routine to be called on detection of errors.
81e5c89e4eSSatish Balay 
82e5c89e4eSSatish Balay    Not Collective
83e5c89e4eSSatish Balay 
84e5c89e4eSSatish Balay    Input Parameters:
85e5c89e4eSSatish Balay +  handler - error handler routine
86e5c89e4eSSatish Balay -  ctx - optional handler context that contains information needed by the handler (for
87e5c89e4eSSatish Balay          example file pointers for error messages etc.)
88e5c89e4eSSatish Balay 
89e5c89e4eSSatish Balay    Calling sequence of handler:
90efca3c55SSatish Balay $    int handler(MPI_Comm comm,int line,char *func,char *file,PetscErrorCode n,int p,char *mess,void *ctx);
91e5c89e4eSSatish Balay 
92a5b23f4aSJose E. Roman +  comm - communicator over which error occurred
93e5c89e4eSSatish Balay .  line - the line number of the error (indicated by __LINE__)
94e5c89e4eSSatish Balay .  file - the file in which the error was detected (indicated by __FILE__)
95e5c89e4eSSatish Balay .  n - the generic error number (see list defined in include/petscerror.h)
96668f157eSBarry Smith .  p - PETSC_ERROR_INITIAL if error just detected, otherwise PETSC_ERROR_REPEAT
97e5c89e4eSSatish Balay .  mess - an error text string, usually just printed to the screen
98e5c89e4eSSatish Balay -  ctx - the error handler context
99e5c89e4eSSatish Balay 
100e5c89e4eSSatish Balay    Options Database Keys:
10110699b91SBarry Smith +   -on_error_attach_debugger <noxterm,gdb or dbx> - starts up the debugger if an error occurs
10210699b91SBarry Smith -   -on_error_abort - aborts the program if an error occurs
103e5c89e4eSSatish Balay 
104e5c89e4eSSatish Balay    Level: intermediate
105e5c89e4eSSatish Balay 
106811af0c4SBarry Smith    Note:
107811af0c4SBarry Smith    The currently available PETSc error handlers include `PetscTraceBackErrorHandler()`,
108811af0c4SBarry Smith    `PetscAttachDebuggerErrorHandler()`, `PetscAbortErrorHandler()`, and `PetscMPIAbortErrorHandler()`, `PetscReturnErrorHandler()`.
109e93bc3c1Svictor 
110811af0c4SBarry Smith    Fortran Note:
11195452b02SPatrick Sanan     You can only push one error handler from Fortran before poping it.
1127850c7c0SBarry Smith 
113db781477SPatrick Sanan .seealso: `PetscPopErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscAbortErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscPushSignalHandler()`
114e5c89e4eSSatish Balay @*/
115d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscPushErrorHandler(PetscErrorCode (*handler)(MPI_Comm comm, int, const char *, const char *, PetscErrorCode, PetscErrorType, const char *, void *), void *ctx)
116d71ae5a4SJacob Faibussowitsch {
117e5c89e4eSSatish Balay   EH neweh;
118e5c89e4eSSatish Balay 
119e5c89e4eSSatish Balay   PetscFunctionBegin;
1209566063dSJacob Faibussowitsch   PetscCall(PetscNew(&neweh));
121a297a907SKarl Rupp   if (eh) neweh->previous = eh;
12202c9f0b5SLisandro Dalcin   else neweh->previous = NULL;
123e5c89e4eSSatish Balay   neweh->handler = handler;
124e5c89e4eSSatish Balay   neweh->ctx     = ctx;
125e5c89e4eSSatish Balay   eh             = neweh;
126e5c89e4eSSatish Balay   PetscFunctionReturn(0);
127e5c89e4eSSatish Balay }
128e5c89e4eSSatish Balay 
129e30d2299SSatish Balay /*@
130e5c89e4eSSatish Balay    PetscPopErrorHandler - Removes the latest error handler that was
131811af0c4SBarry Smith    pushed with `PetscPushErrorHandler()`.
132e5c89e4eSSatish Balay 
133e5c89e4eSSatish Balay    Not Collective
134e5c89e4eSSatish Balay 
135e5c89e4eSSatish Balay    Level: intermediate
136e5c89e4eSSatish Balay 
137db781477SPatrick Sanan .seealso: `PetscPushErrorHandler()`
138e5c89e4eSSatish Balay @*/
139d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscPopErrorHandler(void)
140d71ae5a4SJacob Faibussowitsch {
141e5c89e4eSSatish Balay   EH tmp;
142e5c89e4eSSatish Balay 
143e5c89e4eSSatish Balay   PetscFunctionBegin;
144e5c89e4eSSatish Balay   if (!eh) PetscFunctionReturn(0);
145e5c89e4eSSatish Balay   tmp = eh;
146e5c89e4eSSatish Balay   eh  = eh->previous;
1479566063dSJacob Faibussowitsch   PetscCall(PetscFree(tmp));
148e5c89e4eSSatish Balay   PetscFunctionReturn(0);
149e5c89e4eSSatish Balay }
150e5c89e4eSSatish Balay 
151e93bc3c1Svictor /*@C
15245b666d6SBarry Smith   PetscReturnErrorHandler - Error handler that causes a return without printing an error message.
153e93bc3c1Svictor 
154e93bc3c1Svictor    Not Collective
155e93bc3c1Svictor 
156e93bc3c1Svictor    Input Parameters:
157e32f2f54SBarry Smith +  comm - communicator over which error occurred
158e32f2f54SBarry Smith .  line - the line number of the error (indicated by __LINE__)
159e93bc3c1Svictor .  file - the file in which the error was detected (indicated by __FILE__)
160e93bc3c1Svictor .  mess - an error text string, usually just printed to the screen
161e93bc3c1Svictor .  n - the generic error number
162e93bc3c1Svictor .  p - specific error number
163e93bc3c1Svictor -  ctx - error handler context
164e93bc3c1Svictor 
165e93bc3c1Svictor    Level: developer
166e93bc3c1Svictor 
167e93bc3c1Svictor    Notes:
168e93bc3c1Svictor    Most users need not directly employ this routine and the other error
169811af0c4SBarry Smith    handlers, but can instead use the simplified interface `SETERRQ()`, which has
170e93bc3c1Svictor    the calling sequence
171e32f2f54SBarry Smith $     SETERRQ(comm,number,mess)
172e93bc3c1Svictor 
173811af0c4SBarry Smith    `PetscIgnoreErrorHandler()` does the same thing as this function, but is deprecated, you should use this function.
174e93bc3c1Svictor 
175811af0c4SBarry Smith    Use `PetscPushErrorHandler()` to set the desired error handler.
176e93bc3c1Svictor 
177db781477SPatrick Sanan .seealso: `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscError()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`, `PetscTraceBackErrorHandler()`,
178db781477SPatrick Sanan           `PetscAttachDebuggerErrorHandler()`, `PetscEmacsClientErrorHandler()`
179e93bc3c1Svictor  @*/
180d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscReturnErrorHandler(MPI_Comm comm, int line, const char *fun, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, void *ctx)
181d71ae5a4SJacob Faibussowitsch {
182362febeeSStefano Zampini   return n;
183e93bc3c1Svictor }
184e93bc3c1Svictor 
185e5c89e4eSSatish Balay static char PetscErrorBaseMessage[1024];
186e5c89e4eSSatish Balay /*
187e5c89e4eSSatish Balay        The numerical values for these are defined in include/petscerror.h; any changes
188e5c89e4eSSatish Balay    there must also be made here
189e5c89e4eSSatish Balay */
190e5c89e4eSSatish Balay static const char *PetscErrorStrings[] = {
191e5c89e4eSSatish Balay   /*55 */ "Out of memory",
192e5c89e4eSSatish Balay   "No support for this operation for this object type",
193e5c89e4eSSatish Balay   "No support for this operation on this system",
194e5c89e4eSSatish Balay   /*58 */ "Operation done in wrong order",
195e5c89e4eSSatish Balay   /*59 */ "Signal received",
196e5c89e4eSSatish Balay   /*60 */ "Nonconforming object sizes",
197e5c89e4eSSatish Balay   "Argument aliasing not permitted",
198e5c89e4eSSatish Balay   "Invalid argument",
199e5c89e4eSSatish Balay   /*63 */ "Argument out of range",
200a17b96a8SKyle Gerard Felker   "Corrupt argument: https://petsc.org/release/faq/#valgrind",
201e5c89e4eSSatish Balay   "Unable to open file",
202e5c89e4eSSatish Balay   "Read from file failed",
203e5c89e4eSSatish Balay   "Write to file failed",
204e5c89e4eSSatish Balay   "Invalid pointer",
205e5c89e4eSSatish Balay   /*69 */ "Arguments must have same type",
206a8b45ee7SBarry Smith   /*70 */ "Attempt to use a pointer that does not point to a valid accessible location",
207a17b96a8SKyle Gerard Felker   /*71 */ "Zero pivot in LU factorization: https://petsc.org/release/faq/#zeropivot",
208e5c89e4eSSatish Balay   /*72 */ "Floating point exception",
209e5c89e4eSSatish Balay   /*73 */ "Object is in wrong state",
210e5c89e4eSSatish Balay   "Corrupted Petsc object",
211e5c89e4eSSatish Balay   "Arguments are incompatible",
212e5c89e4eSSatish Balay   "Error in external library",
213e5c89e4eSSatish Balay   /*77 */ "Petsc has generated inconsistent data",
214a17b96a8SKyle Gerard Felker   "Memory corruption: https://petsc.org/release/faq/#valgrind",
215e5c89e4eSSatish Balay   "Unexpected data in file",
216e5c89e4eSSatish Balay   /*80 */ "Arguments must have same communicators",
217a17b96a8SKyle Gerard Felker   /*81 */ "Zero pivot in Cholesky factorization: https://petsc.org/release/faq/#zeropivot",
218e5c89e4eSSatish Balay   "",
219e5c89e4eSSatish Balay   "",
220a17b96a8SKyle Gerard Felker   "Overflow in integer operation: https://petsc.org/release/faq/#64-bit-indices",
221e5c89e4eSSatish Balay   /*85 */ "Null argument, when expecting valid pointer",
222a17b96a8SKyle Gerard Felker   /*86 */ "Unknown type. Check for miss-spelling or missing package: https://petsc.org/release/install/install/#external-packages",
2233d96e996SBarry Smith   /*87 */ "MPI library at runtime is not compatible with MPI used at compile time",
2248cda6cd7SBarry Smith   /*88 */ "Error in system call",
225a17b96a8SKyle Gerard Felker   /*89 */ "Object Type not set: https://petsc.org/release/faq/#object-type-not-set",
22673260a9bSLisandro Dalcin   /*90 */ "",
22773260a9bSLisandro Dalcin   /*   */ "",
228a17b96a8SKyle Gerard Felker   /*92 */ "See https://petsc.org/release/overview/linear_solve_table/ for possible LU and Cholesky solvers",
229f560318cSPatrick Sanan   /*93 */ "You cannot overwrite this option since that will conflict with other previously set options",
230691b26d3SBarry Smith   /*94 */ "Example/application run with number of MPI ranks it does not support",
231691b26d3SBarry Smith   /*95 */ "Missing or incorrect user input",
232e57d7714SBarry Smith   /*96 */ "GPU resources unavailable",
233764761abSStefano Zampini   /*97 */ "GPU error",
234e809461dSJacob Faibussowitsch   /*98 */ "General MPI error",
235e809461dSJacob Faibussowitsch   /*99 */ "PetscError() incorrectly returned an error code of 0"};
236e5c89e4eSSatish Balay 
237e5c89e4eSSatish Balay /*@C
238e5c89e4eSSatish Balay    PetscErrorMessage - returns the text string associated with a PETSc error code.
239e5c89e4eSSatish Balay 
240e5c89e4eSSatish Balay    Not Collective
241e5c89e4eSSatish Balay 
242e5c89e4eSSatish Balay    Input Parameter:
243e5c89e4eSSatish Balay .   errnum - the error code
244e5c89e4eSSatish Balay 
245d8d19677SJose E. Roman    Output Parameters:
2460298fd71SBarry Smith +  text - the error message (NULL if not desired)
247811af0c4SBarry Smith -  specific - the specific error message that was set with `SETERRQ()` or `PetscError()`.  (NULL if not desired)
248e5c89e4eSSatish Balay 
249e5c89e4eSSatish Balay    Level: developer
250e5c89e4eSSatish Balay 
251db781477SPatrick Sanan .seealso: `PetscPushErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscError()`, `SETERRQ()`, `PetscCall()`
252db781477SPatrick Sanan           `PetscAbortErrorHandler()`, `PetscTraceBackErrorHandler()`
253e5c89e4eSSatish Balay  @*/
254d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscErrorMessage(int errnum, const char *text[], char **specific)
255d71ae5a4SJacob Faibussowitsch {
256c35ec7c7SPierre Jolivet   size_t len;
257c35ec7c7SPierre Jolivet 
258e5c89e4eSSatish Balay   PetscFunctionBegin;
259c35ec7c7SPierre Jolivet   if (text && errnum > PETSC_ERR_MIN_VALUE && errnum < PETSC_ERR_MAX_VALUE) {
260c35ec7c7SPierre Jolivet     *text = PetscErrorStrings[errnum - PETSC_ERR_MIN_VALUE - 1];
261c35ec7c7SPierre Jolivet     PetscCall(PetscStrlen(*text, &len));
262c35ec7c7SPierre Jolivet     if (!len) *text = NULL;
2639371c9d4SSatish Balay   } else if (text) *text = NULL;
264e5c89e4eSSatish Balay 
265a297a907SKarl Rupp   if (specific) *specific = PetscErrorBaseMessage;
266e5c89e4eSSatish Balay   PetscFunctionReturn(0);
267e5c89e4eSSatish Balay }
268e5c89e4eSSatish Balay 
269984a1229SMatthew G. Knepley #if defined(PETSC_CLANGUAGE_CXX)
270984a1229SMatthew G. Knepley   /* C++ exceptions are formally not allowed to propagate through extern "C" code. In practice, far too much software
271984a1229SMatthew G. Knepley  * would be broken if implementations did not handle it it some common cases. However, keep in mind
272984a1229SMatthew G. Knepley  *
273984a1229SMatthew G. Knepley  *   Rule 62. Don't allow exceptions to propagate across module boundaries
274984a1229SMatthew G. Knepley  *
275984a1229SMatthew G. Knepley  * in "C++ Coding Standards" by Sutter and Alexandrescu. (This accounts for part of the ongoing C++ binary interface
276984a1229SMatthew G. Knepley  * instability.) Having PETSc raise errors as C++ exceptions was probably misguided and should eventually be removed.
277984a1229SMatthew G. Knepley  *
278984a1229SMatthew G. Knepley  * Here is the problem: You have a C++ function call a PETSc function, and you would like to maintain the error message
279984a1229SMatthew G. Knepley  * and stack information from the PETSc error. You could make everyone write exactly this code in their C++, but that
280984a1229SMatthew G. Knepley  * seems crazy to me.
281984a1229SMatthew G. Knepley  */
282984a1229SMatthew G. Knepley   #include <sstream>
2834c94c282SMatthew G. Knepley   #include <stdexcept>
284d71ae5a4SJacob Faibussowitsch static void PetscCxxErrorThrow()
285d71ae5a4SJacob Faibussowitsch {
286984a1229SMatthew G. Knepley   const char *str;
287984a1229SMatthew G. Knepley   if (eh && eh->ctx) {
288984a1229SMatthew G. Knepley     std::ostringstream *msg;
289984a1229SMatthew G. Knepley     msg = (std::ostringstream *)eh->ctx;
290984a1229SMatthew G. Knepley     str = msg->str().c_str();
291984a1229SMatthew G. Knepley   } else str = "Error detected in C PETSc";
292984a1229SMatthew G. Knepley 
293984a1229SMatthew G. Knepley   throw std::runtime_error(str);
294984a1229SMatthew G. Knepley }
295984a1229SMatthew G. Knepley #endif
296984a1229SMatthew G. Knepley 
297e5c89e4eSSatish Balay /*@C
29845b666d6SBarry Smith    PetscError - Routine that is called when an error has been detected, usually called through the macro SETERRQ(PETSC_COMM_SELF,).
299e5c89e4eSSatish Balay 
300811af0c4SBarry Smith   Collective
301e5c89e4eSSatish Balay 
302e5c89e4eSSatish Balay    Input Parameters:
303e32f2f54SBarry Smith +  comm - communicator over which error occurred.  ALL ranks of this communicator MUST call this routine
304e32f2f54SBarry Smith .  line - the line number of the error (indicated by __LINE__)
3053de71b31SHong Zhang .  func - the function name in which the error was detected
306e5c89e4eSSatish Balay .  file - the file in which the error was detected (indicated by __FILE__)
307e5c89e4eSSatish Balay .  n - the generic error number
308811af0c4SBarry Smith .  p - `PETSC_ERROR_INITIAL` indicates the error was initially detected, `PETSC_ERROR_REPEAT` indicates this is a traceback from a previously detected error
309e5c89e4eSSatish Balay -  mess - formatted message string - aka printf
310e5c89e4eSSatish Balay 
311811af0c4SBarry Smith   Options Database Keys:
31245b666d6SBarry Smith +  -error_output_stdout - output the error messages to stdout instead of the default stderr
31345b666d6SBarry Smith -  -error_output_none - do not output the error messages
31445b666d6SBarry Smith 
315e5c89e4eSSatish Balay   Level: intermediate
316e5c89e4eSSatish Balay 
317e5c89e4eSSatish Balay    Notes:
318e809461dSJacob Faibussowitsch    PETSc error handling is done with error return codes. A non-zero return indicates an error
319e809461dSJacob Faibussowitsch    was detected. The return-value of this routine is what is ultimately returned by
320e809461dSJacob Faibussowitsch    `SETERRQ()`.
32145b666d6SBarry Smith 
322e809461dSJacob Faibussowitsch    Note that numerical errors (potential divide by zero, for example) are not managed by the
323e809461dSJacob Faibussowitsch    error return codes; they are managed via, for example, `KSPGetConvergedReason()` that
324e809461dSJacob Faibussowitsch    indicates if the solve was successful or not. The option `-ksp_error_if_not_converged`, for
325e809461dSJacob Faibussowitsch    example, turns numerical failures into hard errors managed via `PetscError()`.
32645b666d6SBarry Smith 
327e809461dSJacob Faibussowitsch    PETSc provides a rich supply of error handlers, see the list below, and users can also
328e809461dSJacob Faibussowitsch    provide their own error handlers.
329811af0c4SBarry Smith 
330e809461dSJacob Faibussowitsch    If the user sets their own error handler (via `PetscPushErrorHandler()`) they may return any
331e809461dSJacob Faibussowitsch    arbitrary value from it, but are encouraged to return nonzero values. If the return value is
332e809461dSJacob Faibussowitsch    zero, `SETERRQ()` will ignore the value and return `PETSC_ERR_RETURN` (a nonzero value)
333e809461dSJacob Faibussowitsch    instead.
334e809461dSJacob Faibussowitsch 
335e809461dSJacob Faibussowitsch    Most users need not directly use this routine and the error handlers, but can instead use
336e809461dSJacob Faibussowitsch    the simplified interface `PetscCall()` or `SETERRQ()`.
337e5c89e4eSSatish Balay 
338e3081792SBarry Smith    Fortran Note:
339e3081792SBarry Smith    This routine is used differently from Fortran
340e3081792SBarry Smith $    PetscError(MPI_Comm comm,PetscErrorCode n,PetscErrorType p,char *message)
341e3081792SBarry Smith 
342811af0c4SBarry Smith    Developer Note:
343811af0c4SBarry Smith    Since this is called after an error condition it should not be calling any error handlers (currently it ignores any error codes)
3443bf036e2SBarry Smith    BUT this routine does call regular PETSc functions that may call error handlers, this is problematic and could be fixed by never calling other PETSc routines
3453bf036e2SBarry Smith    but this annoying.
3463bf036e2SBarry Smith 
347db781477SPatrick Sanan .seealso: `PetscErrorCode`, `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`,
348db781477SPatrick Sanan           `PetscReturnErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscEmacsClientErrorHandler()`,
349db781477SPatrick Sanan           `SETERRQ()`, `PetscCall()`, `CHKMEMQ`, `SETERRQ()`, `SETERRQ()`, `PetscErrorMessage()`, `PETSCABORT()`
350e5c89e4eSSatish Balay @*/
351d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscError(MPI_Comm comm, int line, const char *func, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, ...)
352d71ae5a4SJacob Faibussowitsch {
353e5c89e4eSSatish Balay   va_list        Argp;
354c9a19010SBarry Smith   size_t         fullLength;
35502c9f0b5SLisandro Dalcin   char           buf[2048], *lbuf = NULL;
356fbfcfee5SBarry Smith   PetscBool      ismain;
3573bf036e2SBarry Smith   PetscErrorCode ierr;
358e5c89e4eSSatish Balay 
35939a651e2SJacob Faibussowitsch   if (!PetscErrorHandlingInitialized) return n;
3600d577829SBarry Smith   if (comm == MPI_COMM_NULL) comm = PETSC_COMM_SELF;
361e5c89e4eSSatish Balay 
362e5c89e4eSSatish Balay   /* Compose the message evaluating the print format */
363e5c89e4eSSatish Balay   if (mess) {
364e5c89e4eSSatish Balay     va_start(Argp, mess);
3652d609e63SMatthew Knepley     PetscVSNPrintf(buf, 2048, mess, &fullLength, Argp);
366e5c89e4eSSatish Balay     va_end(Argp);
367e5c89e4eSSatish Balay     lbuf = buf;
36878179f8bSBarry Smith     if (p == PETSC_ERROR_INITIAL) PetscStrncpy(PetscErrorBaseMessage, lbuf, 1023);
369e5c89e4eSSatish Balay   }
370e5c89e4eSSatish Balay 
3714ed0ab5bSBarry Smith   if (p == PETSC_ERROR_INITIAL && n != PETSC_ERR_MEMC) PetscMallocValidate(__LINE__, PETSC_FUNCTION_NAME, __FILE__);
3724ed0ab5bSBarry Smith 
37302c9f0b5SLisandro Dalcin   if (!eh) ierr = PetscTraceBackErrorHandler(comm, line, func, file, n, p, lbuf, NULL);
374efca3c55SSatish Balay   else ierr = (*eh->handler)(comm, line, func, file, n, p, lbuf, eh->ctx);
375362febeeSStefano Zampini   PetscStackClearTop;
376e5c89e4eSSatish Balay 
377e5c89e4eSSatish Balay   /*
3787233276eSBarry Smith       If this is called from the main() routine we call MPI_Abort() instead of
379e5c89e4eSSatish Balay     return to allow the parallel program to be properly shutdown.
380e5c89e4eSSatish Balay 
3817233276eSBarry Smith     Does not call PETSCABORT() since that would provide the wrong source file and line number information
382e5c89e4eSSatish Balay   */
38349c86fc7SBarry Smith   if (func) {
384e5c89e4eSSatish Balay     PetscStrncmp(func, "main", 4, &ismain);
3857233276eSBarry Smith     if (ismain) {
38649c86fc7SBarry Smith       if (petscwaitonerrorflg) PetscSleep(1000);
387660278c0SBarry Smith       PETSCABORT(comm, ierr);
3887233276eSBarry Smith     }
38949c86fc7SBarry Smith   }
3902c280183SJed Brown #if defined(PETSC_CLANGUAGE_CXX)
391ad540459SPierre Jolivet   if (p == PETSC_ERROR_IN_CXX) PetscCxxErrorThrow();
392d736bfebSBarry Smith #endif
393362febeeSStefano Zampini   return ierr;
394e5c89e4eSSatish Balay }
395e5c89e4eSSatish Balay 
396e5c89e4eSSatish Balay /* -------------------------------------------------------------------------*/
397e5c89e4eSSatish Balay 
398e5c89e4eSSatish Balay /*@C
399e5c89e4eSSatish Balay     PetscIntView - Prints an array of integers; useful for debugging.
400e5c89e4eSSatish Balay 
401811af0c4SBarry Smith     Collective on viewer
402e5c89e4eSSatish Balay 
403e5c89e4eSSatish Balay     Input Parameters:
404e5c89e4eSSatish Balay +   N - number of integers in array
405e5c89e4eSSatish Balay .   idx - array of integers
406811af0c4SBarry Smith -   viewer - location to print array,  `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
407e5c89e4eSSatish Balay 
408e5c89e4eSSatish Balay   Level: intermediate
409e5c89e4eSSatish Balay 
410811af0c4SBarry Smith     Note:
411811af0c4SBarry Smith     This may be called from within the debugger
412300a7f5bSBarry Smith 
413811af0c4SBarry Smith     Developer Note:
414811af0c4SBarry Smith     idx cannot be const because may be passed to binary viewer where temporary byte swapping may be done
415811af0c4SBarry Smith 
416811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscRealView()`
417e5c89e4eSSatish Balay @*/
418d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscIntView(PetscInt N, const PetscInt idx[], PetscViewer viewer)
419d71ae5a4SJacob Faibussowitsch {
420ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
421e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 20, p = N % 20;
422ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
423e5c89e4eSSatish Balay   MPI_Comm    comm;
424e5c89e4eSSatish Balay 
425e5c89e4eSSatish Balay   PetscFunctionBegin;
426e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
4270700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
4285ab33896SBarry Smith   if (N) PetscValidIntPointer(idx, 2);
4299566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
4309566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
4319566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
432e5c89e4eSSatish Balay 
4339566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
4349566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
435e5c89e4eSSatish Balay   if (iascii) {
4369566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
437e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
438ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4399566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * i));
440ca0c3be5SJacob Faibussowitsch       } else {
4419566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * i));
442ca0c3be5SJacob Faibussowitsch       }
44348a46eb9SPierre Jolivet       for (j = 0; j < 20; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[i * 20 + j]));
4449566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
445e5c89e4eSSatish Balay     }
446e5c89e4eSSatish Balay     if (p) {
447ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4489566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * n));
449ca0c3be5SJacob Faibussowitsch       } else {
4509566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * n));
451ca0c3be5SJacob Faibussowitsch       }
4529566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[20 * n + i]));
4539566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
454e5c89e4eSSatish Balay     }
4559566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
4569566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
4576805f65bSBarry Smith   } else if (isbinary) {
458ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
459e5c89e4eSSatish Balay     PetscInt    *array;
460783b601eSJed Brown 
4619566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
462e5c89e4eSSatish Balay 
463e5c89e4eSSatish Balay     if (size > 1) {
464e5c89e4eSSatish Balay       if (rank) {
4659566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
4669566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, NULL, NULL, NULL, MPIU_INT, 0, comm));
467e5c89e4eSSatish Balay       } else {
4689566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
4699566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
470e5c89e4eSSatish Balay         Ntotal = sizes[0];
4719566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
472e5c89e4eSSatish Balay         displs[0] = 0;
473e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
474e5c89e4eSSatish Balay           Ntotal += sizes[i];
475e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
476e5c89e4eSSatish Balay         }
4779566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
4789566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, array, sizes, displs, MPIU_INT, 0, comm));
4799566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_INT));
4809566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
4819566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
4829566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
483e5c89e4eSSatish Balay       }
484e5c89e4eSSatish Balay     } else {
4859566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, idx, N, PETSC_INT));
486e5c89e4eSSatish Balay     }
487e5c89e4eSSatish Balay   } else {
488e5c89e4eSSatish Balay     const char *tname;
4899566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
49098921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
491e5c89e4eSSatish Balay   }
492e5c89e4eSSatish Balay   PetscFunctionReturn(0);
493e5c89e4eSSatish Balay }
494e5c89e4eSSatish Balay 
495e5c89e4eSSatish Balay /*@C
496e5c89e4eSSatish Balay     PetscRealView - Prints an array of doubles; useful for debugging.
497e5c89e4eSSatish Balay 
498811af0c4SBarry Smith     Collective on viewer
499e5c89e4eSSatish Balay 
500e5c89e4eSSatish Balay     Input Parameters:
501811af0c4SBarry Smith +   N - number of `PetscReal` in array
502811af0c4SBarry Smith .   idx - array of `PetscReal`
503811af0c4SBarry Smith -   viewer - location to print array,  `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
504e5c89e4eSSatish Balay 
505e5c89e4eSSatish Balay   Level: intermediate
506e5c89e4eSSatish Balay 
507811af0c4SBarry Smith     Note:
508811af0c4SBarry Smith     This may be called from within the debugger
509300a7f5bSBarry Smith 
510811af0c4SBarry Smith     Developer Note:
511811af0c4SBarry Smith     idx cannot be const because may be passed to binary viewer where temporary byte swapping may be done
512811af0c4SBarry Smith 
513811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`
514e5c89e4eSSatish Balay @*/
515d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscRealView(PetscInt N, const PetscReal idx[], PetscViewer viewer)
516d71ae5a4SJacob Faibussowitsch {
517ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
518e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 5, p = N % 5;
519ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
520e5c89e4eSSatish Balay   MPI_Comm    comm;
521e5c89e4eSSatish Balay 
522e5c89e4eSSatish Balay   PetscFunctionBegin;
523e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
5240700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
525064a246eSJacob Faibussowitsch   PetscValidRealPointer(idx, 2);
5269566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
5279566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
5289566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
529e5c89e4eSSatish Balay 
5309566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
5319566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
532e5c89e4eSSatish Balay   if (iascii) {
5331a989b97SToby Isaac     PetscInt tab;
5341a989b97SToby Isaac 
5359566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
5369566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIGetTab(viewer, &tab));
537e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
5389566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
539ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5409566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * i));
541ca0c3be5SJacob Faibussowitsch       } else {
5429566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * i));
543ca0c3be5SJacob Faibussowitsch       }
5449566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
54548a46eb9SPierre Jolivet       for (j = 0; j < 5; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 5 + j]));
5469566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
547e5c89e4eSSatish Balay     }
548e5c89e4eSSatish Balay     if (p) {
5499566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
550ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5519566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * n));
552ca0c3be5SJacob Faibussowitsch       } else {
5539566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * n));
554ca0c3be5SJacob Faibussowitsch       }
5559566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
5569566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[5 * n + i]));
5579566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
558e5c89e4eSSatish Balay     }
5599566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
5609566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIISetTab(viewer, tab));
5619566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
5626805f65bSBarry Smith   } else if (isbinary) {
563ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, *displs, Ntotal, NN;
564e5c89e4eSSatish Balay     PetscReal   *array;
565e5c89e4eSSatish Balay 
5669566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
567e5c89e4eSSatish Balay 
568e5c89e4eSSatish Balay     if (size > 1) {
569e5c89e4eSSatish Balay       if (rank) {
5709566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
5719566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, NULL, NULL, NULL, MPIU_REAL, 0, comm));
572e5c89e4eSSatish Balay       } else {
5739566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
5749566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
575e5c89e4eSSatish Balay         Ntotal = sizes[0];
5769566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
577e5c89e4eSSatish Balay         displs[0] = 0;
578e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
579e5c89e4eSSatish Balay           Ntotal += sizes[i];
580e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
581e5c89e4eSSatish Balay         }
5829566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
5839566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, array, sizes, displs, MPIU_REAL, 0, comm));
5849566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_REAL));
5859566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
5869566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
5879566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
588e5c89e4eSSatish Balay       }
589e5c89e4eSSatish Balay     } else {
5909566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_REAL));
591e5c89e4eSSatish Balay     }
592e5c89e4eSSatish Balay   } else {
593e5c89e4eSSatish Balay     const char *tname;
5949566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
59598921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
596e5c89e4eSSatish Balay   }
597e5c89e4eSSatish Balay   PetscFunctionReturn(0);
598e5c89e4eSSatish Balay }
599e5c89e4eSSatish Balay 
600e5c89e4eSSatish Balay /*@C
601811af0c4SBarry Smith     PetscScalarView - Prints an array of `PetscScalar`; useful for debugging.
602e5c89e4eSSatish Balay 
603811af0c4SBarry Smith     Collective on viewer
604e5c89e4eSSatish Balay 
605e5c89e4eSSatish Balay     Input Parameters:
606e5c89e4eSSatish Balay +   N - number of scalars in array
607e5c89e4eSSatish Balay .   idx - array of scalars
608811af0c4SBarry Smith -   viewer - location to print array,  `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
609e5c89e4eSSatish Balay 
610e5c89e4eSSatish Balay   Level: intermediate
611e5c89e4eSSatish Balay 
612811af0c4SBarry Smith     Note:
613811af0c4SBarry Smith     This may be called from within the debugger
614300a7f5bSBarry Smith 
615811af0c4SBarry Smith     Developer Note:
616811af0c4SBarry Smith     idx cannot be const because may be passed to binary viewer where byte swapping may be done
617811af0c4SBarry Smith 
618811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`, `PetscRealView()`
619e5c89e4eSSatish Balay @*/
620d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscScalarView(PetscInt N, const PetscScalar idx[], PetscViewer viewer)
621d71ae5a4SJacob Faibussowitsch {
622ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
623e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 3, p = N % 3;
624ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
625e5c89e4eSSatish Balay   MPI_Comm    comm;
626e5c89e4eSSatish Balay 
627e5c89e4eSSatish Balay   PetscFunctionBegin;
628e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
629e5c89e4eSSatish Balay   PetscValidHeader(viewer, 3);
6308c34849bSStefano Zampini   if (N) PetscValidScalarPointer(idx, 2);
6319566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
6329566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
6339566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
634e5c89e4eSSatish Balay 
6359566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
6369566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
637e5c89e4eSSatish Balay   if (iascii) {
6389566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
639e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
640ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6419566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * i));
642ca0c3be5SJacob Faibussowitsch       } else {
6439566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * i));
644ca0c3be5SJacob Faibussowitsch       }
645e5c89e4eSSatish Balay       for (j = 0; j < 3; j++) {
646e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6479566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[i * 3 + j]), (double)PetscImaginaryPart(idx[i * 3 + j])));
648e5c89e4eSSatish Balay #else
6499566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 3 + j]));
650e5c89e4eSSatish Balay #endif
651e5c89e4eSSatish Balay       }
6529566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
653e5c89e4eSSatish Balay     }
654e5c89e4eSSatish Balay     if (p) {
655ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6569566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * n));
657ca0c3be5SJacob Faibussowitsch       } else {
6589566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * n));
659ca0c3be5SJacob Faibussowitsch       }
660e5c89e4eSSatish Balay       for (i = 0; i < p; i++) {
661e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6629566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[n * 3 + i]), (double)PetscImaginaryPart(idx[n * 3 + i])));
663e5c89e4eSSatish Balay #else
6649566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[3 * n + i]));
665e5c89e4eSSatish Balay #endif
666e5c89e4eSSatish Balay       }
6679566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
668e5c89e4eSSatish Balay     }
6699566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
6709566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
6716805f65bSBarry Smith   } else if (isbinary) {
672ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
673e5c89e4eSSatish Balay     PetscScalar *array;
674e5c89e4eSSatish Balay 
6759566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
676e5c89e4eSSatish Balay 
677e5c89e4eSSatish Balay     if (size > 1) {
678e5c89e4eSSatish Balay       if (rank) {
6799566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
6809566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, NULL, NULL, NULL, MPIU_SCALAR, 0, comm));
681e5c89e4eSSatish Balay       } else {
6829566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
6839566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
684e5c89e4eSSatish Balay         Ntotal = sizes[0];
6859566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
686e5c89e4eSSatish Balay         displs[0] = 0;
687e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
688e5c89e4eSSatish Balay           Ntotal += sizes[i];
689e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
690e5c89e4eSSatish Balay         }
6919566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
6929566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, array, sizes, displs, MPIU_SCALAR, 0, comm));
6939566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_SCALAR));
6949566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
6959566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
6969566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
697e5c89e4eSSatish Balay       }
698e5c89e4eSSatish Balay     } else {
6999566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_SCALAR));
700e5c89e4eSSatish Balay     }
701e5c89e4eSSatish Balay   } else {
702e5c89e4eSSatish Balay     const char *tname;
7039566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
70498921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
705e5c89e4eSSatish Balay   }
706e5c89e4eSSatish Balay   PetscFunctionReturn(0);
707e5c89e4eSSatish Balay }
708e5c89e4eSSatish Balay 
709e22e20c5SJunchao Zhang #if defined(PETSC_HAVE_CUDA)
7100e6b6b59SJacob Faibussowitsch   #include <petscdevice_cuda.h>
711d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUBLASGetErrorName(cublasStatus_t status)
712d71ae5a4SJacob Faibussowitsch {
713e22e20c5SJunchao Zhang   switch (status) {
714e22e20c5SJunchao Zhang   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
715d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_SUCCESS:
716d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_SUCCESS";
717d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_INITIALIZED:
718d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_INITIALIZED";
719d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ALLOC_FAILED:
720d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ALLOC_FAILED";
721d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INVALID_VALUE:
722d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INVALID_VALUE";
723d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ARCH_MISMATCH:
724d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ARCH_MISMATCH";
725d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_MAPPING_ERROR:
726d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_MAPPING_ERROR";
727d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_EXECUTION_FAILED:
728d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_EXECUTION_FAILED";
729d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INTERNAL_ERROR:
730d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INTERNAL_ERROR";
731d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_SUPPORTED:
732d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_SUPPORTED";
733d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_LICENSE_ERROR:
734d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_LICENSE_ERROR";
735e22e20c5SJunchao Zhang   #endif
736d71ae5a4SJacob Faibussowitsch   default:
737d71ae5a4SJacob Faibussowitsch     return "unknown error";
738e22e20c5SJunchao Zhang   }
739e22e20c5SJunchao Zhang }
740d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUSolverGetErrorName(cusolverStatus_t status)
741d71ae5a4SJacob Faibussowitsch {
742a4b895e1SBarry Smith   switch (status) {
743a4b895e1SBarry Smith   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
744d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_SUCCESS:
745d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_SUCCESS";
746d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_INITIALIZED:
747d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_INITIALIZED";
748d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_VALUE:
749d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_VALUE";
750d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ARCH_MISMATCH:
751d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ARCH_MISMATCH";
752d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INTERNAL_ERROR:
753d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INTERNAL_ERROR";
754030f984aSJacob Faibussowitsch     #if (CUDART_VERSION >= 9000) /* CUDA 9.0 had these defined on June 2021 */
755d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ALLOC_FAILED:
756d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ALLOC_FAILED";
757d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MAPPING_ERROR:
758d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MAPPING_ERROR";
759d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_EXECUTION_FAILED:
760d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_EXECUTION_FAILED";
761d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
762d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
763d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_SUPPORTED:
764d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_SUPPORTED ";
765d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ZERO_PIVOT:
766d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ZERO_PIVOT";
767d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_LICENSE:
768d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_LICENSE";
769a4b895e1SBarry Smith     #endif
770030f984aSJacob Faibussowitsch   #endif
771d71ae5a4SJacob Faibussowitsch   default:
772d71ae5a4SJacob Faibussowitsch     return "unknown error";
773030f984aSJacob Faibussowitsch   }
774030f984aSJacob Faibussowitsch }
775d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUFFTGetErrorName(cufftResult result)
776d71ae5a4SJacob Faibussowitsch {
777030f984aSJacob Faibussowitsch   switch (result) {
778d71ae5a4SJacob Faibussowitsch   case CUFFT_SUCCESS:
779d71ae5a4SJacob Faibussowitsch     return "CUFFT_SUCCESS";
780d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_PLAN:
781d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_PLAN";
782d71ae5a4SJacob Faibussowitsch   case CUFFT_ALLOC_FAILED:
783d71ae5a4SJacob Faibussowitsch     return "CUFFT_ALLOC_FAILED";
784d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_TYPE:
785d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_TYPE";
786d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_VALUE:
787d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_VALUE";
788d71ae5a4SJacob Faibussowitsch   case CUFFT_INTERNAL_ERROR:
789d71ae5a4SJacob Faibussowitsch     return "CUFFT_INTERNAL_ERROR";
790d71ae5a4SJacob Faibussowitsch   case CUFFT_EXEC_FAILED:
791d71ae5a4SJacob Faibussowitsch     return "CUFFT_EXEC_FAILED";
792d71ae5a4SJacob Faibussowitsch   case CUFFT_SETUP_FAILED:
793d71ae5a4SJacob Faibussowitsch     return "CUFFT_SETUP_FAILED";
794d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_SIZE:
795d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_SIZE";
796d71ae5a4SJacob Faibussowitsch   case CUFFT_UNALIGNED_DATA:
797d71ae5a4SJacob Faibussowitsch     return "CUFFT_UNALIGNED_DATA";
798d71ae5a4SJacob Faibussowitsch   case CUFFT_INCOMPLETE_PARAMETER_LIST:
799d71ae5a4SJacob Faibussowitsch     return "CUFFT_INCOMPLETE_PARAMETER_LIST";
800d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_DEVICE:
801d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_DEVICE";
802d71ae5a4SJacob Faibussowitsch   case CUFFT_PARSE_ERROR:
803d71ae5a4SJacob Faibussowitsch     return "CUFFT_PARSE_ERROR";
804d71ae5a4SJacob Faibussowitsch   case CUFFT_NO_WORKSPACE:
805d71ae5a4SJacob Faibussowitsch     return "CUFFT_NO_WORKSPACE";
806d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_IMPLEMENTED:
807d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_IMPLEMENTED";
808d71ae5a4SJacob Faibussowitsch   case CUFFT_LICENSE_ERROR:
809d71ae5a4SJacob Faibussowitsch     return "CUFFT_LICENSE_ERROR";
810d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_SUPPORTED:
811d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_SUPPORTED";
812d71ae5a4SJacob Faibussowitsch   default:
813d71ae5a4SJacob Faibussowitsch     return "unknown error";
814a4b895e1SBarry Smith   }
815a4b895e1SBarry Smith }
816e22e20c5SJunchao Zhang #endif
81759af0bd3SScott Kruger 
81859af0bd3SScott Kruger #if defined(PETSC_HAVE_HIP)
8190e6b6b59SJacob Faibussowitsch   #include <petscdevice_hip.h>
820d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscHIPBLASGetErrorName(hipblasStatus_t status)
821d71ae5a4SJacob Faibussowitsch {
82259af0bd3SScott Kruger   switch (status) {
823d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_SUCCESS:
824d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_SUCCESS";
825d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_INITIALIZED:
826d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_INITIALIZED";
827d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ALLOC_FAILED:
828d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ALLOC_FAILED";
829d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INVALID_VALUE:
830d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INVALID_VALUE";
831d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ARCH_MISMATCH:
832d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ARCH_MISMATCH";
833d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_MAPPING_ERROR:
834d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_MAPPING_ERROR";
835d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_EXECUTION_FAILED:
836d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_EXECUTION_FAILED";
837d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INTERNAL_ERROR:
838d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INTERNAL_ERROR";
839d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_SUPPORTED:
840d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_SUPPORTED";
841d71ae5a4SJacob Faibussowitsch   default:
842d71ae5a4SJacob Faibussowitsch     return "unknown error";
84359af0bd3SScott Kruger   }
84459af0bd3SScott Kruger }
845*47d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSPARSEGetErrorName(hipsparseStatus_t status)
846*47d993e7Ssuyashtn {
847*47d993e7Ssuyashtn   switch (status) {
848*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_SUCCESS:
849*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_SUCCESS";
850*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_INITIALIZED:
851*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_INITIALIZED";
852*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_ALLOC_FAILED:
853*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ALLOC_FAILED";
854*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_INVALID_VALUE:
855*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INVALID_VALUE";
856*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_ARCH_MISMATCH:
857*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ARCH_MISMATCH";
858*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_MAPPING_ERROR:
859*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MAPPING_ERROR";
860*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_EXECUTION_FAILED:
861*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_EXECUTION_FAILED";
862*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_INTERNAL_ERROR:
863*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INTERNAL_ERROR";
864*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
865*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
866*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_ZERO_PIVOT:
867*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ZERO_PIVOT";
868*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_SUPPORTED:
869*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_SUPPORTED";
870*47d993e7Ssuyashtn   case HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES:
871*47d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES";
872*47d993e7Ssuyashtn   default:
873*47d993e7Ssuyashtn     return "unknown error";
874*47d993e7Ssuyashtn   }
875*47d993e7Ssuyashtn }
876*47d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSolverGetErrorName(hipsolverStatus_t status)
877*47d993e7Ssuyashtn {
878*47d993e7Ssuyashtn   switch (status) {
879*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_SUCCESS:
880*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_SUCCESS";
881*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_INITIALIZED:
882*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_INITIALIZED";
883*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_ALLOC_FAILED:
884*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ALLOC_FAILED";
885*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_MAPPING_ERROR:
886*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_MAPPING_ERROR";
887*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_VALUE:
888*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_VALUE";
889*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_EXECUTION_FAILED:
890*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_EXECUTION_FAILED";
891*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_INTERNAL_ERROR:
892*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INTERNAL_ERROR";
893*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_SUPPORTED:
894*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_SUPPORTED ";
895*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_ARCH_MISMATCH:
896*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ARCH_MISMATCH";
897*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_HANDLE_IS_NULLPTR:
898*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_HANDLE_IS_NULLPTR";
899*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_ENUM:
900*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_ENUM";
901*47d993e7Ssuyashtn   case HIPSOLVER_STATUS_UNKNOWN:
902*47d993e7Ssuyashtn   default:
903*47d993e7Ssuyashtn     return "HIPSOLVER_STATUS_UNKNOWN";
904*47d993e7Ssuyashtn   }
905*47d993e7Ssuyashtn }
90659af0bd3SScott Kruger #endif
907db9cea48SBarry Smith 
908db9cea48SBarry Smith /*@
909811af0c4SBarry Smith       PetscMPIErrorString - Given an MPI error code returns the `MPI_Error_string()` appropriately
910db9cea48SBarry Smith            formatted for displaying with the PETSc error handlers.
911db9cea48SBarry Smith 
912db9cea48SBarry Smith  Input Parameter:
913db9cea48SBarry Smith .  err - the MPI error code
914db9cea48SBarry Smith 
915db9cea48SBarry Smith  Output Parameter:
916811af0c4SBarry Smith .  string - the MPI error message, should declare its length to be larger than `MPI_MAX_ERROR_STRING`
917db9cea48SBarry Smith 
91849c86fc7SBarry Smith    Level: developer
91949c86fc7SBarry Smith 
920811af0c4SBarry Smith  Note:
921db9cea48SBarry Smith     Does not return an error code or do error handling because it may be called from inside an error handler
922db9cea48SBarry Smith 
923db9cea48SBarry Smith @*/
924d71ae5a4SJacob Faibussowitsch void PetscMPIErrorString(PetscMPIInt err, char *string)
925d71ae5a4SJacob Faibussowitsch {
926db9cea48SBarry Smith   char        errorstring[MPI_MAX_ERROR_STRING];
927db9cea48SBarry Smith   PetscMPIInt len, j = 0;
928db9cea48SBarry Smith 
929db9cea48SBarry Smith   MPI_Error_string(err, (char *)errorstring, &len);
930db9cea48SBarry Smith   for (PetscMPIInt i = 0; i < len; i++) {
931db9cea48SBarry Smith     string[j++] = errorstring[i];
932db9cea48SBarry Smith     if (errorstring[i] == '\n') {
933db9cea48SBarry Smith       for (PetscMPIInt k = 0; k < 16; k++) string[j++] = ' ';
934db9cea48SBarry Smith     }
935db9cea48SBarry Smith   }
936db9cea48SBarry Smith   string[j] = 0;
937db9cea48SBarry Smith }
938