xref: /petsc/src/sys/error/err.c (revision 10450e9e44b354a0a3da7bbd573407bdf051df10)
17d0a6c19SBarry Smith 
2e5c89e4eSSatish Balay /*
3e5c89e4eSSatish Balay       Code that allows one to set the error handlers
447d993e7Ssuyashtn       Portions of this code are under:
547d993e7Ssuyashtn       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__)
29*10450e9eSJacob Faibussowitsch . fun  - the function name
30e5c89e4eSSatish Balay . mess - an error text string, usually just printed to the screen
31e5c89e4eSSatish Balay . n    - the generic error number
32e5c89e4eSSatish Balay . p    - specific error number
33e5c89e4eSSatish Balay - ctx  - error handler context
34e5c89e4eSSatish Balay 
35e5c89e4eSSatish Balay   Options Database Key:
3610699b91SBarry Smith . -on_error_emacs <machinename> - will contact machinename to open the Emacs client there
37e5c89e4eSSatish Balay 
38e5c89e4eSSatish Balay   Level: developer
39e5c89e4eSSatish Balay 
40811af0c4SBarry Smith   Note:
41e5c89e4eSSatish Balay   You must put (server-start) in your .emacs file for the emacsclient software to work
42e5c89e4eSSatish Balay 
43aec76313SJacob Faibussowitsch   Developer Notes:
44811af0c4SBarry Smith   Since this is an error handler it cannot call `PetscCall()`; thus we just return if an error is detected.
45811af0c4SBarry Smith   But some of the functions it calls do perform error checking that may not be appropriate in a error handler call.
463bf036e2SBarry Smith 
47db781477SPatrick Sanan .seealso: `PetscError()`, `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscAttachDebuggerErrorHandler()`,
48db781477SPatrick Sanan           `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscReturnErrorHandler()`
49e5c89e4eSSatish Balay  @*/
50d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscEmacsClientErrorHandler(MPI_Comm comm, int line, const char *fun, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, void *ctx)
51d71ae5a4SJacob Faibussowitsch {
52e5c89e4eSSatish Balay   PetscErrorCode ierr;
53e5c89e4eSSatish Balay   char           command[PETSC_MAX_PATH_LEN];
54e5c89e4eSSatish Balay   const char    *pdir;
55e5c89e4eSSatish Balay   FILE          *fp;
56e5c89e4eSSatish Balay 
579371c9d4SSatish Balay   ierr = PetscGetPetscDir(&pdir);
5811cc89d2SBarry Smith   if (ierr) return ierr;
59a364092eSJacob Faibussowitsch   ierr = PetscSNPrintf(command, PETSC_STATIC_ARRAY_LENGTH(command), "cd %s; emacsclient --no-wait +%d %s\n", pdir, line, file);
60a364092eSJacob Faibussowitsch   if (ierr) return ierr;
61e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
629371c9d4SSatish Balay   ierr = PetscPOpen(MPI_COMM_WORLD, (char *)ctx, command, "r", &fp);
6311cc89d2SBarry Smith   if (ierr) return ierr;
649371c9d4SSatish Balay   ierr = PetscPClose(MPI_COMM_WORLD, fp);
6511cc89d2SBarry Smith   if (ierr) return ierr;
66e5c89e4eSSatish Balay #else
67e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP_SYS, "Cannot run external programs on this machine");
68e5c89e4eSSatish Balay #endif
699371c9d4SSatish Balay   ierr = PetscPopErrorHandler();
7011cc89d2SBarry Smith   if (ierr) return ierr; /* remove this handler from the stack of handlers */
713bf036e2SBarry Smith   if (!eh) {
729371c9d4SSatish Balay     ierr = PetscTraceBackErrorHandler(comm, line, fun, file, n, p, mess, NULL);
7311cc89d2SBarry Smith     if (ierr) return ierr;
743bf036e2SBarry Smith   } else {
759371c9d4SSatish Balay     ierr = (*eh->handler)(comm, line, fun, file, n, p, mess, eh->ctx);
7611cc89d2SBarry Smith     if (ierr) return ierr;
773bf036e2SBarry Smith   }
783ba16761SJacob Faibussowitsch   return PETSC_SUCCESS;
79e5c89e4eSSatish Balay }
80e5c89e4eSSatish Balay 
81e5c89e4eSSatish Balay /*@C
82e5c89e4eSSatish Balay   PetscPushErrorHandler - Sets a routine to be called on detection of errors.
83e5c89e4eSSatish Balay 
84e5c89e4eSSatish Balay   Not Collective
85e5c89e4eSSatish Balay 
86e5c89e4eSSatish Balay   Input Parameters:
87e5c89e4eSSatish Balay + handler - error handler routine
88e5c89e4eSSatish Balay - ctx     - optional handler context that contains information needed by the handler (for
89e5c89e4eSSatish Balay             example file pointers for error messages etc.)
90e5c89e4eSSatish Balay 
9120f4b53cSBarry Smith   Calling sequence of `handler`:
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__)
95*10450e9eSJacob Faibussowitsch . fun  - the function name
96e5c89e4eSSatish Balay . n    - the generic error number (see list defined in include/petscerror.h)
9720f4b53cSBarry Smith . p    - `PETSC_ERROR_INITIAL` if error just detected, otherwise `PETSC_ERROR_REPEAT`
98e5c89e4eSSatish Balay . mess - an error text string, usually just printed to the screen
99e5c89e4eSSatish Balay - ctx  - the error handler context
100e5c89e4eSSatish Balay 
101e5c89e4eSSatish Balay   Options Database Keys:
10210699b91SBarry Smith + -on_error_attach_debugger <noxterm,gdb or dbx> - starts up the debugger if an error occurs
10310699b91SBarry Smith - -on_error_abort                                - aborts the program if an error occurs
104e5c89e4eSSatish Balay 
105e5c89e4eSSatish Balay   Level: intermediate
106e5c89e4eSSatish Balay 
107811af0c4SBarry Smith   Note:
108811af0c4SBarry Smith   The currently available PETSc error handlers include `PetscTraceBackErrorHandler()`,
109811af0c4SBarry Smith   `PetscAttachDebuggerErrorHandler()`, `PetscAbortErrorHandler()`, and `PetscMPIAbortErrorHandler()`, `PetscReturnErrorHandler()`.
110e93bc3c1Svictor 
111aec76313SJacob Faibussowitsch   Fortran Notes:
112aaa8cc7dSPierre Jolivet   You can only push one error handler from Fortran before popping it.
1137850c7c0SBarry Smith 
114db781477SPatrick Sanan .seealso: `PetscPopErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscAbortErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscPushSignalHandler()`
115e5c89e4eSSatish Balay @*/
116*10450e9eSJacob Faibussowitsch PetscErrorCode PetscPushErrorHandler(PetscErrorCode (*handler)(MPI_Comm comm, int line, const char *fun, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, void *ctx), void *ctx)
117d71ae5a4SJacob Faibussowitsch {
118e5c89e4eSSatish Balay   EH neweh;
119e5c89e4eSSatish Balay 
120e5c89e4eSSatish Balay   PetscFunctionBegin;
1219566063dSJacob Faibussowitsch   PetscCall(PetscNew(&neweh));
122a297a907SKarl Rupp   if (eh) neweh->previous = eh;
12302c9f0b5SLisandro Dalcin   else neweh->previous = NULL;
124e5c89e4eSSatish Balay   neweh->handler = handler;
125e5c89e4eSSatish Balay   neweh->ctx     = ctx;
126e5c89e4eSSatish Balay   eh             = neweh;
1273ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
128e5c89e4eSSatish Balay }
129e5c89e4eSSatish Balay 
130e30d2299SSatish Balay /*@
131e5c89e4eSSatish Balay   PetscPopErrorHandler - Removes the latest error handler that was
132811af0c4SBarry Smith   pushed with `PetscPushErrorHandler()`.
133e5c89e4eSSatish Balay 
134e5c89e4eSSatish Balay   Not Collective
135e5c89e4eSSatish Balay 
136e5c89e4eSSatish Balay   Level: intermediate
137e5c89e4eSSatish Balay 
138db781477SPatrick Sanan .seealso: `PetscPushErrorHandler()`
139e5c89e4eSSatish Balay @*/
140d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscPopErrorHandler(void)
141d71ae5a4SJacob Faibussowitsch {
142e5c89e4eSSatish Balay   EH tmp;
143e5c89e4eSSatish Balay 
144e5c89e4eSSatish Balay   PetscFunctionBegin;
1453ba16761SJacob Faibussowitsch   if (!eh) PetscFunctionReturn(PETSC_SUCCESS);
146e5c89e4eSSatish Balay   tmp = eh;
147e5c89e4eSSatish Balay   eh  = eh->previous;
1489566063dSJacob Faibussowitsch   PetscCall(PetscFree(tmp));
1493ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
150e5c89e4eSSatish Balay }
151e5c89e4eSSatish Balay 
152e93bc3c1Svictor /*@C
15345b666d6SBarry Smith   PetscReturnErrorHandler - Error handler that causes a return without printing an error message.
154e93bc3c1Svictor 
155e93bc3c1Svictor   Not Collective
156e93bc3c1Svictor 
157e93bc3c1Svictor   Input Parameters:
158e32f2f54SBarry Smith + comm - communicator over which error occurred
159e32f2f54SBarry Smith . line - the line number of the error (indicated by __LINE__)
160*10450e9eSJacob Faibussowitsch . fun  - the function name
161e93bc3c1Svictor . file - the file in which the error was detected (indicated by __FILE__)
162e93bc3c1Svictor . mess - an error text string, usually just printed to the screen
163e93bc3c1Svictor . n    - the generic error number
164e93bc3c1Svictor . p    - specific error number
165e93bc3c1Svictor - ctx  - error handler context
166e93bc3c1Svictor 
167e93bc3c1Svictor   Level: developer
168e93bc3c1Svictor 
169e93bc3c1Svictor   Notes:
170e93bc3c1Svictor   Most users need not directly employ this routine and the other error
171811af0c4SBarry Smith   handlers, but can instead use the simplified interface `SETERRQ()`, which has
172e93bc3c1Svictor   the calling sequence
173e32f2f54SBarry Smith $     SETERRQ(comm, number, mess)
174e93bc3c1Svictor 
175811af0c4SBarry Smith   `PetscIgnoreErrorHandler()` does the same thing as this function, but is deprecated, you should use this function.
176e93bc3c1Svictor 
177811af0c4SBarry Smith   Use `PetscPushErrorHandler()` to set the desired error handler.
178e93bc3c1Svictor 
179db781477SPatrick Sanan .seealso: `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscError()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`, `PetscTraceBackErrorHandler()`,
180db781477SPatrick Sanan           `PetscAttachDebuggerErrorHandler()`, `PetscEmacsClientErrorHandler()`
181e93bc3c1Svictor  @*/
182d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscReturnErrorHandler(MPI_Comm comm, int line, const char *fun, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, void *ctx)
183d71ae5a4SJacob Faibussowitsch {
184*10450e9eSJacob Faibussowitsch   (void)comm;
185*10450e9eSJacob Faibussowitsch   (void)line;
186*10450e9eSJacob Faibussowitsch   (void)fun;
187*10450e9eSJacob Faibussowitsch   (void)file;
188*10450e9eSJacob Faibussowitsch   (void)p;
189*10450e9eSJacob Faibussowitsch   (void)mess;
190*10450e9eSJacob Faibussowitsch   (void)ctx;
191362febeeSStefano Zampini   return n;
192e93bc3c1Svictor }
193e93bc3c1Svictor 
194e5c89e4eSSatish Balay static char PetscErrorBaseMessage[1024];
195e5c89e4eSSatish Balay /*
196e5c89e4eSSatish Balay        The numerical values for these are defined in include/petscerror.h; any changes
197e5c89e4eSSatish Balay    there must also be made here
198e5c89e4eSSatish Balay */
199e5c89e4eSSatish Balay static const char *PetscErrorStrings[] = {
200e5c89e4eSSatish Balay   /*55 */ "Out of memory",
201e5c89e4eSSatish Balay   "No support for this operation for this object type",
202e5c89e4eSSatish Balay   "No support for this operation on this system",
203e5c89e4eSSatish Balay   /*58 */ "Operation done in wrong order",
204e5c89e4eSSatish Balay   /*59 */ "Signal received",
205e5c89e4eSSatish Balay   /*60 */ "Nonconforming object sizes",
206e5c89e4eSSatish Balay   "Argument aliasing not permitted",
207e5c89e4eSSatish Balay   "Invalid argument",
208e5c89e4eSSatish Balay   /*63 */ "Argument out of range",
209a17b96a8SKyle Gerard Felker   "Corrupt argument: https://petsc.org/release/faq/#valgrind",
210e5c89e4eSSatish Balay   "Unable to open file",
211e5c89e4eSSatish Balay   "Read from file failed",
212e5c89e4eSSatish Balay   "Write to file failed",
213e5c89e4eSSatish Balay   "Invalid pointer",
214e5c89e4eSSatish Balay   /*69 */ "Arguments must have same type",
215a8b45ee7SBarry Smith   /*70 */ "Attempt to use a pointer that does not point to a valid accessible location",
216a17b96a8SKyle Gerard Felker   /*71 */ "Zero pivot in LU factorization: https://petsc.org/release/faq/#zeropivot",
217e5c89e4eSSatish Balay   /*72 */ "Floating point exception",
218e5c89e4eSSatish Balay   /*73 */ "Object is in wrong state",
219e5c89e4eSSatish Balay   "Corrupted Petsc object",
220e5c89e4eSSatish Balay   "Arguments are incompatible",
221e5c89e4eSSatish Balay   "Error in external library",
222e5c89e4eSSatish Balay   /*77 */ "Petsc has generated inconsistent data",
223a17b96a8SKyle Gerard Felker   "Memory corruption: https://petsc.org/release/faq/#valgrind",
224e5c89e4eSSatish Balay   "Unexpected data in file",
225e5c89e4eSSatish Balay   /*80 */ "Arguments must have same communicators",
226a17b96a8SKyle Gerard Felker   /*81 */ "Zero pivot in Cholesky factorization: https://petsc.org/release/faq/#zeropivot",
227e5c89e4eSSatish Balay   "",
228e5c89e4eSSatish Balay   "",
229a17b96a8SKyle Gerard Felker   "Overflow in integer operation: https://petsc.org/release/faq/#64-bit-indices",
230e5c89e4eSSatish Balay   /*85 */ "Null argument, when expecting valid pointer",
231a17b96a8SKyle Gerard Felker   /*86 */ "Unknown type. Check for miss-spelling or missing package: https://petsc.org/release/install/install/#external-packages",
2323d96e996SBarry Smith   /*87 */ "MPI library at runtime is not compatible with MPI used at compile time",
2338cda6cd7SBarry Smith   /*88 */ "Error in system call",
234a17b96a8SKyle Gerard Felker   /*89 */ "Object Type not set: https://petsc.org/release/faq/#object-type-not-set",
23573260a9bSLisandro Dalcin   /*90 */ "",
23673260a9bSLisandro Dalcin   /*   */ "",
237a17b96a8SKyle Gerard Felker   /*92 */ "See https://petsc.org/release/overview/linear_solve_table/ for possible LU and Cholesky solvers",
238f560318cSPatrick Sanan   /*93 */ "You cannot overwrite this option since that will conflict with other previously set options",
239691b26d3SBarry Smith   /*94 */ "Example/application run with number of MPI ranks it does not support",
240691b26d3SBarry Smith   /*95 */ "Missing or incorrect user input",
241e57d7714SBarry Smith   /*96 */ "GPU resources unavailable",
242764761abSStefano Zampini   /*97 */ "GPU error",
243e809461dSJacob Faibussowitsch   /*98 */ "General MPI error",
244e809461dSJacob Faibussowitsch   /*99 */ "PetscError() incorrectly returned an error code of 0"};
245e5c89e4eSSatish Balay 
246e5c89e4eSSatish Balay /*@C
2473ba16761SJacob Faibussowitsch   PetscErrorMessage - Returns the text string associated with a PETSc error code.
248e5c89e4eSSatish Balay 
249e5c89e4eSSatish Balay   Not Collective
250e5c89e4eSSatish Balay 
251e5c89e4eSSatish Balay   Input Parameter:
252e5c89e4eSSatish Balay . errnum - the error code
253e5c89e4eSSatish Balay 
254d8d19677SJose E. Roman   Output Parameters:
2553ba16761SJacob Faibussowitsch + text     - the error message (`NULL` if not desired)
2563ba16761SJacob Faibussowitsch - specific - the specific error message that was set with `SETERRQ()` or
2573ba16761SJacob Faibussowitsch              `PetscError()`. (`NULL` if not desired)
258e5c89e4eSSatish Balay 
259e5c89e4eSSatish Balay   Level: developer
260e5c89e4eSSatish Balay 
2613ba16761SJacob Faibussowitsch .seealso: `PetscErrorCode`, `PetscPushErrorHandler()`, `PetscAttachDebuggerErrorHandler()`,
2623ba16761SJacob Faibussowitsch `PetscError()`, `SETERRQ()`, `PetscCall()` `PetscAbortErrorHandler()`,
2633ba16761SJacob Faibussowitsch `PetscTraceBackErrorHandler()`
264e5c89e4eSSatish Balay @*/
2653ba16761SJacob Faibussowitsch PetscErrorCode PetscErrorMessage(PetscErrorCode errnum, const char *text[], char **specific)
266d71ae5a4SJacob Faibussowitsch {
2673ba16761SJacob Faibussowitsch   PetscFunctionBegin;
2683ba16761SJacob Faibussowitsch   if (text) {
2693ba16761SJacob Faibussowitsch     if (errnum > PETSC_ERR_MIN_VALUE && errnum < PETSC_ERR_MAX_VALUE) {
270c35ec7c7SPierre Jolivet       size_t len;
271c35ec7c7SPierre Jolivet 
272c35ec7c7SPierre Jolivet       *text = PetscErrorStrings[errnum - PETSC_ERR_MIN_VALUE - 1];
273c35ec7c7SPierre Jolivet       PetscCall(PetscStrlen(*text, &len));
274c35ec7c7SPierre Jolivet       if (!len) *text = NULL;
2753ba16761SJacob Faibussowitsch     } else if (errnum == PETSC_ERR_BOOLEAN_MACRO_FAILURE) {
2763ba16761SJacob Faibussowitsch       /* this "error code" arises from failures in boolean macros, where the || operator is
2773ba16761SJacob Faibussowitsch          used to short-circuit the macro call in case of error. This has the side effect of
2783ba16761SJacob Faibussowitsch          "returning" either 0 (PETSC_SUCCESS) or 1 (PETSC_ERR_UNKNONWN):
279e5c89e4eSSatish Balay 
2803ba16761SJacob Faibussowitsch          #define PETSC_FOO(x) ((PetscErrorCode)(PetscBar(x) || PetscBaz(x)))
2813ba16761SJacob Faibussowitsch 
2823ba16761SJacob Faibussowitsch          If PetscBar() fails (returns nonzero) PetscBaz() is not executed but the result of
2833ba16761SJacob Faibussowitsch          this expression is boolean false, hence PETSC_ERR_UNNOWN
2843ba16761SJacob Faibussowitsch        */
2853ba16761SJacob Faibussowitsch       *text = "Error occurred in boolean shortcuit in macro";
2863ba16761SJacob Faibussowitsch     } else {
2873ba16761SJacob Faibussowitsch       *text = NULL;
2883ba16761SJacob Faibussowitsch     }
2893ba16761SJacob Faibussowitsch   }
290a297a907SKarl Rupp   if (specific) *specific = PetscErrorBaseMessage;
2913ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
292e5c89e4eSSatish Balay }
293e5c89e4eSSatish Balay 
294984a1229SMatthew G. Knepley #if defined(PETSC_CLANGUAGE_CXX)
295984a1229SMatthew G. Knepley   /* C++ exceptions are formally not allowed to propagate through extern "C" code. In practice, far too much software
296984a1229SMatthew G. Knepley  * would be broken if implementations did not handle it it some common cases. However, keep in mind
297984a1229SMatthew G. Knepley  *
298984a1229SMatthew G. Knepley  *   Rule 62. Don't allow exceptions to propagate across module boundaries
299984a1229SMatthew G. Knepley  *
300984a1229SMatthew G. Knepley  * in "C++ Coding Standards" by Sutter and Alexandrescu. (This accounts for part of the ongoing C++ binary interface
301984a1229SMatthew G. Knepley  * instability.) Having PETSc raise errors as C++ exceptions was probably misguided and should eventually be removed.
302984a1229SMatthew G. Knepley  *
303984a1229SMatthew G. Knepley  * Here is the problem: You have a C++ function call a PETSc function, and you would like to maintain the error message
304984a1229SMatthew G. Knepley  * and stack information from the PETSc error. You could make everyone write exactly this code in their C++, but that
305984a1229SMatthew G. Knepley  * seems crazy to me.
306984a1229SMatthew G. Knepley  */
307984a1229SMatthew G. Knepley   #include <sstream>
3084c94c282SMatthew G. Knepley   #include <stdexcept>
309d71ae5a4SJacob Faibussowitsch static void PetscCxxErrorThrow()
310d71ae5a4SJacob Faibussowitsch {
311984a1229SMatthew G. Knepley   const char *str;
312984a1229SMatthew G. Knepley   if (eh && eh->ctx) {
313984a1229SMatthew G. Knepley     std::ostringstream *msg;
314984a1229SMatthew G. Knepley     msg = (std::ostringstream *)eh->ctx;
315984a1229SMatthew G. Knepley     str = msg->str().c_str();
316984a1229SMatthew G. Knepley   } else str = "Error detected in C PETSc";
317984a1229SMatthew G. Knepley 
318984a1229SMatthew G. Knepley   throw std::runtime_error(str);
319984a1229SMatthew G. Knepley }
320984a1229SMatthew G. Knepley #endif
321984a1229SMatthew G. Knepley 
322e5c89e4eSSatish Balay /*@C
32345b666d6SBarry Smith   PetscError - Routine that is called when an error has been detected, usually called through the macro SETERRQ(PETSC_COMM_SELF,).
324e5c89e4eSSatish Balay 
325811af0c4SBarry Smith   Collective
326e5c89e4eSSatish Balay 
327e5c89e4eSSatish Balay   Input Parameters:
328e32f2f54SBarry Smith + comm - communicator over which error occurred.  ALL ranks of this communicator MUST call this routine
329e32f2f54SBarry Smith . line - the line number of the error (indicated by __LINE__)
3303de71b31SHong Zhang . func - the function name in which the error was detected
331e5c89e4eSSatish Balay . file - the file in which the error was detected (indicated by __FILE__)
332e5c89e4eSSatish Balay . n    - the generic error number
333811af0c4SBarry Smith . p    - `PETSC_ERROR_INITIAL` indicates the error was initially detected, `PETSC_ERROR_REPEAT` indicates this is a traceback from a previously detected error
334e5c89e4eSSatish Balay - mess - formatted message string - aka printf
335e5c89e4eSSatish Balay 
336811af0c4SBarry Smith   Options Database Keys:
33745b666d6SBarry Smith + -error_output_stdout - output the error messages to stdout instead of the default stderr
33845b666d6SBarry Smith - -error_output_none   - do not output the error messages
33945b666d6SBarry Smith 
340e5c89e4eSSatish Balay   Level: intermediate
341e5c89e4eSSatish Balay 
342e5c89e4eSSatish Balay   Notes:
343e809461dSJacob Faibussowitsch   PETSc error handling is done with error return codes. A non-zero return indicates an error
344e809461dSJacob Faibussowitsch   was detected. The return-value of this routine is what is ultimately returned by
345e809461dSJacob Faibussowitsch   `SETERRQ()`.
34645b666d6SBarry Smith 
347e809461dSJacob Faibussowitsch   Note that numerical errors (potential divide by zero, for example) are not managed by the
348e809461dSJacob Faibussowitsch   error return codes; they are managed via, for example, `KSPGetConvergedReason()` that
349e809461dSJacob Faibussowitsch   indicates if the solve was successful or not. The option `-ksp_error_if_not_converged`, for
350e809461dSJacob Faibussowitsch   example, turns numerical failures into hard errors managed via `PetscError()`.
35145b666d6SBarry Smith 
352e809461dSJacob Faibussowitsch   PETSc provides a rich supply of error handlers, see the list below, and users can also
353e809461dSJacob Faibussowitsch   provide their own error handlers.
354811af0c4SBarry Smith 
355e809461dSJacob Faibussowitsch   If the user sets their own error handler (via `PetscPushErrorHandler()`) they may return any
356e809461dSJacob Faibussowitsch   arbitrary value from it, but are encouraged to return nonzero values. If the return value is
357e809461dSJacob Faibussowitsch   zero, `SETERRQ()` will ignore the value and return `PETSC_ERR_RETURN` (a nonzero value)
358e809461dSJacob Faibussowitsch   instead.
359e809461dSJacob Faibussowitsch 
360e809461dSJacob Faibussowitsch   Most users need not directly use this routine and the error handlers, but can instead use
361e809461dSJacob Faibussowitsch   the simplified interface `PetscCall()` or `SETERRQ()`.
362e5c89e4eSSatish Balay 
363aec76313SJacob Faibussowitsch   Fortran Notes:
364e3081792SBarry Smith   This routine is used differently from Fortran
365e3081792SBarry Smith $    PetscError(MPI_Comm comm, PetscErrorCode n, PetscErrorType p, char *message)
366e3081792SBarry Smith 
367aec76313SJacob Faibussowitsch   Developer Notes:
368811af0c4SBarry Smith   Since this is called after an error condition it should not be calling any error handlers (currently it ignores any error codes)
3693bf036e2SBarry 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
3703bf036e2SBarry Smith   but this annoying.
3713bf036e2SBarry Smith 
372db781477SPatrick Sanan .seealso: `PetscErrorCode`, `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`,
373db781477SPatrick Sanan           `PetscReturnErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscEmacsClientErrorHandler()`,
374aec76313SJacob Faibussowitsch           `SETERRQ()`, `PetscCall()`, `CHKMEMQ`, `PetscErrorMessage()`, `PETSCABORT()`
375e5c89e4eSSatish Balay @*/
376d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscError(MPI_Comm comm, int line, const char *func, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, ...)
377d71ae5a4SJacob Faibussowitsch {
378e5c89e4eSSatish Balay   va_list        Argp;
379c9a19010SBarry Smith   size_t         fullLength;
38002c9f0b5SLisandro Dalcin   char           buf[2048], *lbuf = NULL;
381fbfcfee5SBarry Smith   PetscBool      ismain;
3823bf036e2SBarry Smith   PetscErrorCode ierr;
383e5c89e4eSSatish Balay 
38439a651e2SJacob Faibussowitsch   if (!PetscErrorHandlingInitialized) return n;
3850d577829SBarry Smith   if (comm == MPI_COMM_NULL) comm = PETSC_COMM_SELF;
386e5c89e4eSSatish Balay 
387e5c89e4eSSatish Balay   /* Compose the message evaluating the print format */
388e5c89e4eSSatish Balay   if (mess) {
389e5c89e4eSSatish Balay     va_start(Argp, mess);
3903ba16761SJacob Faibussowitsch     ierr = PetscVSNPrintf(buf, 2048, mess, &fullLength, Argp);
391e5c89e4eSSatish Balay     va_end(Argp);
392e5c89e4eSSatish Balay     lbuf = buf;
3939b15cf9aSJacob Faibussowitsch     if (p == PETSC_ERROR_INITIAL) ierr = PetscStrncpy(PetscErrorBaseMessage, lbuf, sizeof(PetscErrorBaseMessage));
394e5c89e4eSSatish Balay   }
395e5c89e4eSSatish Balay 
3963ba16761SJacob Faibussowitsch   if (p == PETSC_ERROR_INITIAL && n != PETSC_ERR_MEMC) ierr = PetscMallocValidate(__LINE__, PETSC_FUNCTION_NAME, __FILE__);
3974ed0ab5bSBarry Smith 
39802c9f0b5SLisandro Dalcin   if (!eh) ierr = PetscTraceBackErrorHandler(comm, line, func, file, n, p, lbuf, NULL);
399efca3c55SSatish Balay   else ierr = (*eh->handler)(comm, line, func, file, n, p, lbuf, eh->ctx);
400362febeeSStefano Zampini   PetscStackClearTop;
401e5c89e4eSSatish Balay 
402e5c89e4eSSatish Balay   /*
403989c49a2SBarry Smith       If this is called from the main() routine we abort the program.
404989c49a2SBarry Smith       We cannot just return because them some MPI processes may continue to attempt to run
405989c49a2SBarry Smith       while this process simply exits.
406e5c89e4eSSatish Balay   */
40749c86fc7SBarry Smith   if (func) {
4083ba16761SJacob Faibussowitsch     PetscErrorCode cmp_ierr = PetscStrncmp(func, "main", 4, &ismain);
4097233276eSBarry Smith     if (ismain) {
4103ba16761SJacob Faibussowitsch       if (petscwaitonerrorflg) cmp_ierr = PetscSleep(1000);
4113ba16761SJacob Faibussowitsch       (void)cmp_ierr;
412660278c0SBarry Smith       PETSCABORT(comm, ierr);
4137233276eSBarry Smith     }
41449c86fc7SBarry Smith   }
4152c280183SJed Brown #if defined(PETSC_CLANGUAGE_CXX)
416ad540459SPierre Jolivet   if (p == PETSC_ERROR_IN_CXX) PetscCxxErrorThrow();
417d736bfebSBarry Smith #endif
418362febeeSStefano Zampini   return ierr;
419e5c89e4eSSatish Balay }
420e5c89e4eSSatish Balay 
421e5c89e4eSSatish Balay /* -------------------------------------------------------------------------*/
422e5c89e4eSSatish Balay 
423e5c89e4eSSatish Balay /*@C
424e5c89e4eSSatish Balay   PetscIntView - Prints an array of integers; useful for debugging.
425e5c89e4eSSatish Balay 
426c3339decSBarry Smith   Collective
427e5c89e4eSSatish Balay 
428e5c89e4eSSatish Balay   Input Parameters:
429e5c89e4eSSatish Balay + N      - number of integers in array
430e5c89e4eSSatish Balay . idx    - array of integers
431811af0c4SBarry Smith - viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
432e5c89e4eSSatish Balay 
433e5c89e4eSSatish Balay   Level: intermediate
434e5c89e4eSSatish Balay 
435811af0c4SBarry Smith   Note:
436811af0c4SBarry Smith   This may be called from within the debugger
437300a7f5bSBarry Smith 
438aec76313SJacob Faibussowitsch   Developer Notes:
439811af0c4SBarry Smith   idx cannot be const because may be passed to binary viewer where temporary byte swapping may be done
440811af0c4SBarry Smith 
441811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscRealView()`
442e5c89e4eSSatish Balay @*/
443d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscIntView(PetscInt N, const PetscInt idx[], PetscViewer viewer)
444d71ae5a4SJacob Faibussowitsch {
445ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
446e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 20, p = N % 20;
447ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
448e5c89e4eSSatish Balay   MPI_Comm    comm;
449e5c89e4eSSatish Balay 
450e5c89e4eSSatish Balay   PetscFunctionBegin;
451e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
4520700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
4535ab33896SBarry Smith   if (N) PetscValidIntPointer(idx, 2);
4549566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
4559566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
4569566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
457e5c89e4eSSatish Balay 
4589566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
4599566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
460e5c89e4eSSatish Balay   if (iascii) {
4619566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
462e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
463ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4649566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * i));
465ca0c3be5SJacob Faibussowitsch       } else {
4669566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * i));
467ca0c3be5SJacob Faibussowitsch       }
46848a46eb9SPierre Jolivet       for (j = 0; j < 20; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[i * 20 + j]));
4699566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
470e5c89e4eSSatish Balay     }
471e5c89e4eSSatish Balay     if (p) {
472ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4739566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * n));
474ca0c3be5SJacob Faibussowitsch       } else {
4759566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * n));
476ca0c3be5SJacob Faibussowitsch       }
4779566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[20 * n + i]));
4789566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
479e5c89e4eSSatish Balay     }
4809566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
4819566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
4826805f65bSBarry Smith   } else if (isbinary) {
483ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
484e5c89e4eSSatish Balay     PetscInt    *array;
485783b601eSJed Brown 
4869566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
487e5c89e4eSSatish Balay 
488e5c89e4eSSatish Balay     if (size > 1) {
489e5c89e4eSSatish Balay       if (rank) {
4909566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
4919566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, NULL, NULL, NULL, MPIU_INT, 0, comm));
492e5c89e4eSSatish Balay       } else {
4939566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
4949566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
495e5c89e4eSSatish Balay         Ntotal = sizes[0];
4969566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
497e5c89e4eSSatish Balay         displs[0] = 0;
498e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
499e5c89e4eSSatish Balay           Ntotal += sizes[i];
500e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
501e5c89e4eSSatish Balay         }
5029566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
5039566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, array, sizes, displs, MPIU_INT, 0, comm));
5049566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_INT));
5059566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
5069566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
5079566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
508e5c89e4eSSatish Balay       }
509e5c89e4eSSatish Balay     } else {
5109566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, idx, N, PETSC_INT));
511e5c89e4eSSatish Balay     }
512e5c89e4eSSatish Balay   } else {
513e5c89e4eSSatish Balay     const char *tname;
5149566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
51598921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
516e5c89e4eSSatish Balay   }
5173ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
518e5c89e4eSSatish Balay }
519e5c89e4eSSatish Balay 
520e5c89e4eSSatish Balay /*@C
521e5c89e4eSSatish Balay   PetscRealView - Prints an array of doubles; useful for debugging.
522e5c89e4eSSatish Balay 
523c3339decSBarry Smith   Collective
524e5c89e4eSSatish Balay 
525e5c89e4eSSatish Balay   Input Parameters:
526811af0c4SBarry Smith + N      - number of `PetscReal` in array
527811af0c4SBarry Smith . idx    - array of `PetscReal`
528811af0c4SBarry Smith - viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
529e5c89e4eSSatish Balay 
530e5c89e4eSSatish Balay   Level: intermediate
531e5c89e4eSSatish Balay 
532811af0c4SBarry Smith   Note:
533811af0c4SBarry Smith   This may be called from within the debugger
534300a7f5bSBarry Smith 
535aec76313SJacob Faibussowitsch   Developer Notes:
536811af0c4SBarry Smith   idx cannot be const because may be passed to binary viewer where temporary byte swapping may be done
537811af0c4SBarry Smith 
538811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`
539e5c89e4eSSatish Balay @*/
540d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscRealView(PetscInt N, const PetscReal idx[], PetscViewer viewer)
541d71ae5a4SJacob Faibussowitsch {
542ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
543e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 5, p = N % 5;
544ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
545e5c89e4eSSatish Balay   MPI_Comm    comm;
546e5c89e4eSSatish Balay 
547e5c89e4eSSatish Balay   PetscFunctionBegin;
548e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
5490700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
550064a246eSJacob Faibussowitsch   PetscValidRealPointer(idx, 2);
5519566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
5529566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
5539566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
554e5c89e4eSSatish Balay 
5559566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
5569566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
557e5c89e4eSSatish Balay   if (iascii) {
5581a989b97SToby Isaac     PetscInt tab;
5591a989b97SToby Isaac 
5609566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
5619566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIGetTab(viewer, &tab));
562e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
5639566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
564ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5659566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * i));
566ca0c3be5SJacob Faibussowitsch       } else {
5679566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * i));
568ca0c3be5SJacob Faibussowitsch       }
5699566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
57048a46eb9SPierre Jolivet       for (j = 0; j < 5; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 5 + j]));
5719566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
572e5c89e4eSSatish Balay     }
573e5c89e4eSSatish Balay     if (p) {
5749566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
575ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5769566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * n));
577ca0c3be5SJacob Faibussowitsch       } else {
5789566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * n));
579ca0c3be5SJacob Faibussowitsch       }
5809566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
5819566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[5 * n + i]));
5829566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
583e5c89e4eSSatish Balay     }
5849566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
5859566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIISetTab(viewer, tab));
5869566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
5876805f65bSBarry Smith   } else if (isbinary) {
588ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, *displs, Ntotal, NN;
589e5c89e4eSSatish Balay     PetscReal   *array;
590e5c89e4eSSatish Balay 
5919566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
592e5c89e4eSSatish Balay 
593e5c89e4eSSatish Balay     if (size > 1) {
594e5c89e4eSSatish Balay       if (rank) {
5959566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
5969566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, NULL, NULL, NULL, MPIU_REAL, 0, comm));
597e5c89e4eSSatish Balay       } else {
5989566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
5999566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
600e5c89e4eSSatish Balay         Ntotal = sizes[0];
6019566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
602e5c89e4eSSatish Balay         displs[0] = 0;
603e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
604e5c89e4eSSatish Balay           Ntotal += sizes[i];
605e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
606e5c89e4eSSatish Balay         }
6079566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
6089566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, array, sizes, displs, MPIU_REAL, 0, comm));
6099566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_REAL));
6109566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
6119566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
6129566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
613e5c89e4eSSatish Balay       }
614e5c89e4eSSatish Balay     } else {
6159566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_REAL));
616e5c89e4eSSatish Balay     }
617e5c89e4eSSatish Balay   } else {
618e5c89e4eSSatish Balay     const char *tname;
6199566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
62098921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
621e5c89e4eSSatish Balay   }
6223ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
623e5c89e4eSSatish Balay }
624e5c89e4eSSatish Balay 
625e5c89e4eSSatish Balay /*@C
626811af0c4SBarry Smith   PetscScalarView - Prints an array of `PetscScalar`; useful for debugging.
627e5c89e4eSSatish Balay 
628c3339decSBarry Smith   Collective
629e5c89e4eSSatish Balay 
630e5c89e4eSSatish Balay   Input Parameters:
631e5c89e4eSSatish Balay + N      - number of scalars in array
632e5c89e4eSSatish Balay . idx    - array of scalars
633811af0c4SBarry Smith - viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
634e5c89e4eSSatish Balay 
635e5c89e4eSSatish Balay   Level: intermediate
636e5c89e4eSSatish Balay 
637811af0c4SBarry Smith   Note:
638811af0c4SBarry Smith   This may be called from within the debugger
639300a7f5bSBarry Smith 
640aec76313SJacob Faibussowitsch   Developer Notes:
641811af0c4SBarry Smith   idx cannot be const because may be passed to binary viewer where byte swapping may be done
642811af0c4SBarry Smith 
643811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`, `PetscRealView()`
644e5c89e4eSSatish Balay @*/
645d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscScalarView(PetscInt N, const PetscScalar idx[], PetscViewer viewer)
646d71ae5a4SJacob Faibussowitsch {
647ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
648e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 3, p = N % 3;
649ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
650e5c89e4eSSatish Balay   MPI_Comm    comm;
651e5c89e4eSSatish Balay 
652e5c89e4eSSatish Balay   PetscFunctionBegin;
653e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
654e5c89e4eSSatish Balay   PetscValidHeader(viewer, 3);
6558c34849bSStefano Zampini   if (N) PetscValidScalarPointer(idx, 2);
6569566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
6579566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
6589566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
659e5c89e4eSSatish Balay 
6609566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
6619566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
662e5c89e4eSSatish Balay   if (iascii) {
6639566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
664e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
665ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6669566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * i));
667ca0c3be5SJacob Faibussowitsch       } else {
6689566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * i));
669ca0c3be5SJacob Faibussowitsch       }
670e5c89e4eSSatish Balay       for (j = 0; j < 3; j++) {
671e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6729566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[i * 3 + j]), (double)PetscImaginaryPart(idx[i * 3 + j])));
673e5c89e4eSSatish Balay #else
6749566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 3 + j]));
675e5c89e4eSSatish Balay #endif
676e5c89e4eSSatish Balay       }
6779566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
678e5c89e4eSSatish Balay     }
679e5c89e4eSSatish Balay     if (p) {
680ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6819566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * n));
682ca0c3be5SJacob Faibussowitsch       } else {
6839566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * n));
684ca0c3be5SJacob Faibussowitsch       }
685e5c89e4eSSatish Balay       for (i = 0; i < p; i++) {
686e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6879566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[n * 3 + i]), (double)PetscImaginaryPart(idx[n * 3 + i])));
688e5c89e4eSSatish Balay #else
6899566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[3 * n + i]));
690e5c89e4eSSatish Balay #endif
691e5c89e4eSSatish Balay       }
6929566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
693e5c89e4eSSatish Balay     }
6949566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
6959566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
6966805f65bSBarry Smith   } else if (isbinary) {
697ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
698e5c89e4eSSatish Balay     PetscScalar *array;
699e5c89e4eSSatish Balay 
7009566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
701e5c89e4eSSatish Balay 
702e5c89e4eSSatish Balay     if (size > 1) {
703e5c89e4eSSatish Balay       if (rank) {
7049566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
7059566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, NULL, NULL, NULL, MPIU_SCALAR, 0, comm));
706e5c89e4eSSatish Balay       } else {
7079566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
7089566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
709e5c89e4eSSatish Balay         Ntotal = sizes[0];
7109566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
711e5c89e4eSSatish Balay         displs[0] = 0;
712e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
713e5c89e4eSSatish Balay           Ntotal += sizes[i];
714e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
715e5c89e4eSSatish Balay         }
7169566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
7179566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, array, sizes, displs, MPIU_SCALAR, 0, comm));
7189566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_SCALAR));
7199566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
7209566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
7219566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
722e5c89e4eSSatish Balay       }
723e5c89e4eSSatish Balay     } else {
7249566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_SCALAR));
725e5c89e4eSSatish Balay     }
726e5c89e4eSSatish Balay   } else {
727e5c89e4eSSatish Balay     const char *tname;
7289566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
72998921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
730e5c89e4eSSatish Balay   }
7313ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
732e5c89e4eSSatish Balay }
733e5c89e4eSSatish Balay 
734e22e20c5SJunchao Zhang #if defined(PETSC_HAVE_CUDA)
7350e6b6b59SJacob Faibussowitsch   #include <petscdevice_cuda.h>
736d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUBLASGetErrorName(cublasStatus_t status)
737d71ae5a4SJacob Faibussowitsch {
738e22e20c5SJunchao Zhang   switch (status) {
739e22e20c5SJunchao Zhang   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
740d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_SUCCESS:
741d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_SUCCESS";
742d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_INITIALIZED:
743d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_INITIALIZED";
744d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ALLOC_FAILED:
745d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ALLOC_FAILED";
746d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INVALID_VALUE:
747d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INVALID_VALUE";
748d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ARCH_MISMATCH:
749d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ARCH_MISMATCH";
750d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_MAPPING_ERROR:
751d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_MAPPING_ERROR";
752d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_EXECUTION_FAILED:
753d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_EXECUTION_FAILED";
754d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INTERNAL_ERROR:
755d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INTERNAL_ERROR";
756d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_SUPPORTED:
757d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_SUPPORTED";
758d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_LICENSE_ERROR:
759d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_LICENSE_ERROR";
760e22e20c5SJunchao Zhang   #endif
761d71ae5a4SJacob Faibussowitsch   default:
762d71ae5a4SJacob Faibussowitsch     return "unknown error";
763e22e20c5SJunchao Zhang   }
764e22e20c5SJunchao Zhang }
765d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUSolverGetErrorName(cusolverStatus_t status)
766d71ae5a4SJacob Faibussowitsch {
767a4b895e1SBarry Smith   switch (status) {
768a4b895e1SBarry Smith   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
769d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_SUCCESS:
770d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_SUCCESS";
771d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_INITIALIZED:
772d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_INITIALIZED";
773d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_VALUE:
774d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_VALUE";
775d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ARCH_MISMATCH:
776d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ARCH_MISMATCH";
777d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INTERNAL_ERROR:
778d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INTERNAL_ERROR";
779030f984aSJacob Faibussowitsch     #if (CUDART_VERSION >= 9000) /* CUDA 9.0 had these defined on June 2021 */
780d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ALLOC_FAILED:
781d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ALLOC_FAILED";
782d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MAPPING_ERROR:
783d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MAPPING_ERROR";
784d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_EXECUTION_FAILED:
785d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_EXECUTION_FAILED";
786d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
787d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
788d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_SUPPORTED:
789d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_SUPPORTED ";
790d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ZERO_PIVOT:
791d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ZERO_PIVOT";
792d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_LICENSE:
793d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_LICENSE";
794a4b895e1SBarry Smith     #endif
795030f984aSJacob Faibussowitsch   #endif
796d71ae5a4SJacob Faibussowitsch   default:
797d71ae5a4SJacob Faibussowitsch     return "unknown error";
798030f984aSJacob Faibussowitsch   }
799030f984aSJacob Faibussowitsch }
800d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUFFTGetErrorName(cufftResult result)
801d71ae5a4SJacob Faibussowitsch {
802030f984aSJacob Faibussowitsch   switch (result) {
803d71ae5a4SJacob Faibussowitsch   case CUFFT_SUCCESS:
804d71ae5a4SJacob Faibussowitsch     return "CUFFT_SUCCESS";
805d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_PLAN:
806d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_PLAN";
807d71ae5a4SJacob Faibussowitsch   case CUFFT_ALLOC_FAILED:
808d71ae5a4SJacob Faibussowitsch     return "CUFFT_ALLOC_FAILED";
809d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_TYPE:
810d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_TYPE";
811d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_VALUE:
812d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_VALUE";
813d71ae5a4SJacob Faibussowitsch   case CUFFT_INTERNAL_ERROR:
814d71ae5a4SJacob Faibussowitsch     return "CUFFT_INTERNAL_ERROR";
815d71ae5a4SJacob Faibussowitsch   case CUFFT_EXEC_FAILED:
816d71ae5a4SJacob Faibussowitsch     return "CUFFT_EXEC_FAILED";
817d71ae5a4SJacob Faibussowitsch   case CUFFT_SETUP_FAILED:
818d71ae5a4SJacob Faibussowitsch     return "CUFFT_SETUP_FAILED";
819d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_SIZE:
820d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_SIZE";
821d71ae5a4SJacob Faibussowitsch   case CUFFT_UNALIGNED_DATA:
822d71ae5a4SJacob Faibussowitsch     return "CUFFT_UNALIGNED_DATA";
823d71ae5a4SJacob Faibussowitsch   case CUFFT_INCOMPLETE_PARAMETER_LIST:
824d71ae5a4SJacob Faibussowitsch     return "CUFFT_INCOMPLETE_PARAMETER_LIST";
825d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_DEVICE:
826d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_DEVICE";
827d71ae5a4SJacob Faibussowitsch   case CUFFT_PARSE_ERROR:
828d71ae5a4SJacob Faibussowitsch     return "CUFFT_PARSE_ERROR";
829d71ae5a4SJacob Faibussowitsch   case CUFFT_NO_WORKSPACE:
830d71ae5a4SJacob Faibussowitsch     return "CUFFT_NO_WORKSPACE";
831d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_IMPLEMENTED:
832d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_IMPLEMENTED";
833d71ae5a4SJacob Faibussowitsch   case CUFFT_LICENSE_ERROR:
834d71ae5a4SJacob Faibussowitsch     return "CUFFT_LICENSE_ERROR";
835d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_SUPPORTED:
836d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_SUPPORTED";
837d71ae5a4SJacob Faibussowitsch   default:
838d71ae5a4SJacob Faibussowitsch     return "unknown error";
839a4b895e1SBarry Smith   }
840a4b895e1SBarry Smith }
841e22e20c5SJunchao Zhang #endif
84259af0bd3SScott Kruger 
84359af0bd3SScott Kruger #if defined(PETSC_HAVE_HIP)
8440e6b6b59SJacob Faibussowitsch   #include <petscdevice_hip.h>
845d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscHIPBLASGetErrorName(hipblasStatus_t status)
846d71ae5a4SJacob Faibussowitsch {
84759af0bd3SScott Kruger   switch (status) {
848d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_SUCCESS:
849d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_SUCCESS";
850d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_INITIALIZED:
851d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_INITIALIZED";
852d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ALLOC_FAILED:
853d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ALLOC_FAILED";
854d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INVALID_VALUE:
855d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INVALID_VALUE";
856d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ARCH_MISMATCH:
857d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ARCH_MISMATCH";
858d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_MAPPING_ERROR:
859d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_MAPPING_ERROR";
860d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_EXECUTION_FAILED:
861d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_EXECUTION_FAILED";
862d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INTERNAL_ERROR:
863d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INTERNAL_ERROR";
864d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_SUPPORTED:
865d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_SUPPORTED";
866d71ae5a4SJacob Faibussowitsch   default:
867d71ae5a4SJacob Faibussowitsch     return "unknown error";
86859af0bd3SScott Kruger   }
86959af0bd3SScott Kruger }
87047d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSPARSEGetErrorName(hipsparseStatus_t status)
87147d993e7Ssuyashtn {
87247d993e7Ssuyashtn   switch (status) {
87347d993e7Ssuyashtn   case HIPSPARSE_STATUS_SUCCESS:
87447d993e7Ssuyashtn     return "HIPSPARSE_STATUS_SUCCESS";
87547d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_INITIALIZED:
87647d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_INITIALIZED";
87747d993e7Ssuyashtn   case HIPSPARSE_STATUS_ALLOC_FAILED:
87847d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ALLOC_FAILED";
87947d993e7Ssuyashtn   case HIPSPARSE_STATUS_INVALID_VALUE:
88047d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INVALID_VALUE";
88147d993e7Ssuyashtn   case HIPSPARSE_STATUS_ARCH_MISMATCH:
88247d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ARCH_MISMATCH";
88347d993e7Ssuyashtn   case HIPSPARSE_STATUS_MAPPING_ERROR:
88447d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MAPPING_ERROR";
88547d993e7Ssuyashtn   case HIPSPARSE_STATUS_EXECUTION_FAILED:
88647d993e7Ssuyashtn     return "HIPSPARSE_STATUS_EXECUTION_FAILED";
88747d993e7Ssuyashtn   case HIPSPARSE_STATUS_INTERNAL_ERROR:
88847d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INTERNAL_ERROR";
88947d993e7Ssuyashtn   case HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
89047d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
89147d993e7Ssuyashtn   case HIPSPARSE_STATUS_ZERO_PIVOT:
89247d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ZERO_PIVOT";
89347d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_SUPPORTED:
89447d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_SUPPORTED";
89547d993e7Ssuyashtn   case HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES:
89647d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES";
89747d993e7Ssuyashtn   default:
89847d993e7Ssuyashtn     return "unknown error";
89947d993e7Ssuyashtn   }
90047d993e7Ssuyashtn }
90147d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSolverGetErrorName(hipsolverStatus_t status)
90247d993e7Ssuyashtn {
90347d993e7Ssuyashtn   switch (status) {
90447d993e7Ssuyashtn   case HIPSOLVER_STATUS_SUCCESS:
90547d993e7Ssuyashtn     return "HIPSOLVER_STATUS_SUCCESS";
90647d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_INITIALIZED:
90747d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_INITIALIZED";
90847d993e7Ssuyashtn   case HIPSOLVER_STATUS_ALLOC_FAILED:
90947d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ALLOC_FAILED";
91047d993e7Ssuyashtn   case HIPSOLVER_STATUS_MAPPING_ERROR:
91147d993e7Ssuyashtn     return "HIPSOLVER_STATUS_MAPPING_ERROR";
91247d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_VALUE:
91347d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_VALUE";
91447d993e7Ssuyashtn   case HIPSOLVER_STATUS_EXECUTION_FAILED:
91547d993e7Ssuyashtn     return "HIPSOLVER_STATUS_EXECUTION_FAILED";
91647d993e7Ssuyashtn   case HIPSOLVER_STATUS_INTERNAL_ERROR:
91747d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INTERNAL_ERROR";
91847d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_SUPPORTED:
91947d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_SUPPORTED ";
92047d993e7Ssuyashtn   case HIPSOLVER_STATUS_ARCH_MISMATCH:
92147d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ARCH_MISMATCH";
92247d993e7Ssuyashtn   case HIPSOLVER_STATUS_HANDLE_IS_NULLPTR:
92347d993e7Ssuyashtn     return "HIPSOLVER_STATUS_HANDLE_IS_NULLPTR";
92447d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_ENUM:
92547d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_ENUM";
92647d993e7Ssuyashtn   case HIPSOLVER_STATUS_UNKNOWN:
92747d993e7Ssuyashtn   default:
92847d993e7Ssuyashtn     return "HIPSOLVER_STATUS_UNKNOWN";
92947d993e7Ssuyashtn   }
93047d993e7Ssuyashtn }
93159af0bd3SScott Kruger #endif
932db9cea48SBarry Smith 
933aec76313SJacob Faibussowitsch /*@C
934811af0c4SBarry Smith   PetscMPIErrorString - Given an MPI error code returns the `MPI_Error_string()` appropriately
935db9cea48SBarry Smith   formatted for displaying with the PETSc error handlers.
936db9cea48SBarry Smith 
937db9cea48SBarry Smith   Input Parameter:
938db9cea48SBarry Smith . err - the MPI error code
939db9cea48SBarry Smith 
940db9cea48SBarry Smith   Output Parameter:
941811af0c4SBarry Smith . string - the MPI error message, should declare its length to be larger than `MPI_MAX_ERROR_STRING`
942db9cea48SBarry Smith 
94349c86fc7SBarry Smith   Level: developer
94449c86fc7SBarry Smith 
945811af0c4SBarry Smith   Note:
946db9cea48SBarry Smith   Does not return an error code or do error handling because it may be called from inside an error handler
947db9cea48SBarry Smith 
948*10450e9eSJacob Faibussowitsch .seealso: `PetscErrorCode` `PetscErrorMessage()`
949db9cea48SBarry Smith @*/
950d71ae5a4SJacob Faibussowitsch void PetscMPIErrorString(PetscMPIInt err, char *string)
951d71ae5a4SJacob Faibussowitsch {
952db9cea48SBarry Smith   char        errorstring[MPI_MAX_ERROR_STRING];
953db9cea48SBarry Smith   PetscMPIInt len, j = 0;
954db9cea48SBarry Smith 
955db9cea48SBarry Smith   MPI_Error_string(err, (char *)errorstring, &len);
956db9cea48SBarry Smith   for (PetscMPIInt i = 0; i < len; i++) {
957db9cea48SBarry Smith     string[j++] = errorstring[i];
958db9cea48SBarry Smith     if (errorstring[i] == '\n') {
959db9cea48SBarry Smith       for (PetscMPIInt k = 0; k < 16; k++) string[j++] = ' ';
960db9cea48SBarry Smith     }
961db9cea48SBarry Smith   }
962db9cea48SBarry Smith   string[j] = 0;
963db9cea48SBarry Smith }
964