xref: /petsc/src/sys/error/err.c (revision 20f4b53cbb5e9bd9ef12b76a8697d60d197cda17)
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__)
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;
58a364092eSJacob Faibussowitsch   ierr = PetscSNPrintf(command, PETSC_STATIC_ARRAY_LENGTH(command), "cd %s; emacsclient --no-wait +%d %s\n", pdir, line, file);
59a364092eSJacob Faibussowitsch   if (ierr) return ierr;
60e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
619371c9d4SSatish Balay   ierr = PetscPOpen(MPI_COMM_WORLD, (char *)ctx, command, "r", &fp);
6211cc89d2SBarry Smith   if (ierr) return ierr;
639371c9d4SSatish Balay   ierr = PetscPClose(MPI_COMM_WORLD, fp);
6411cc89d2SBarry Smith   if (ierr) return ierr;
65e5c89e4eSSatish Balay #else
66e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP_SYS, "Cannot run external programs on this machine");
67e5c89e4eSSatish Balay #endif
689371c9d4SSatish Balay   ierr = PetscPopErrorHandler();
6911cc89d2SBarry Smith   if (ierr) return ierr; /* remove this handler from the stack of handlers */
703bf036e2SBarry Smith   if (!eh) {
719371c9d4SSatish Balay     ierr = PetscTraceBackErrorHandler(comm, line, fun, file, n, p, mess, NULL);
7211cc89d2SBarry Smith     if (ierr) return ierr;
733bf036e2SBarry Smith   } else {
749371c9d4SSatish Balay     ierr = (*eh->handler)(comm, line, fun, file, n, p, mess, eh->ctx);
7511cc89d2SBarry Smith     if (ierr) return ierr;
763bf036e2SBarry Smith   }
773ba16761SJacob Faibussowitsch   return PETSC_SUCCESS;
78e5c89e4eSSatish Balay }
79e5c89e4eSSatish Balay 
80e5c89e4eSSatish Balay /*@C
81e5c89e4eSSatish Balay    PetscPushErrorHandler - Sets a routine to be called on detection of errors.
82e5c89e4eSSatish Balay 
83e5c89e4eSSatish Balay    Not Collective
84e5c89e4eSSatish Balay 
85e5c89e4eSSatish Balay    Input Parameters:
86e5c89e4eSSatish Balay +  handler - error handler routine
87e5c89e4eSSatish Balay -  ctx - optional handler context that contains information needed by the handler (for
88e5c89e4eSSatish Balay          example file pointers for error messages etc.)
89e5c89e4eSSatish Balay 
90*20f4b53cSBarry Smith    Calling sequence of `handler`:
91*20f4b53cSBarry Smith $  PetscErrorCode handler(MPI_Comm comm, int line, char *func, char *file, PetscErrorCode n, int p, char *mess, void *ctx);
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)
96*20f4b53cSBarry 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;
1263ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
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;
1443ba16761SJacob Faibussowitsch   if (!eh) PetscFunctionReturn(PETSC_SUCCESS);
145e5c89e4eSSatish Balay   tmp = eh;
146e5c89e4eSSatish Balay   eh  = eh->previous;
1479566063dSJacob Faibussowitsch   PetscCall(PetscFree(tmp));
1483ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
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
2383ba16761SJacob Faibussowitsch   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:
2463ba16761SJacob Faibussowitsch + text - the error message (`NULL` if not desired)
2473ba16761SJacob Faibussowitsch - specific - the specific error message that was set with `SETERRQ()` or
2483ba16761SJacob Faibussowitsch              `PetscError()`. (`NULL` if not desired)
249e5c89e4eSSatish Balay 
250e5c89e4eSSatish Balay   Level: developer
251e5c89e4eSSatish Balay 
2523ba16761SJacob Faibussowitsch .seealso: `PetscErrorCode`, `PetscPushErrorHandler()`, `PetscAttachDebuggerErrorHandler()`,
2533ba16761SJacob Faibussowitsch `PetscError()`, `SETERRQ()`, `PetscCall()` `PetscAbortErrorHandler()`,
2543ba16761SJacob Faibussowitsch `PetscTraceBackErrorHandler()`
255e5c89e4eSSatish Balay @*/
2563ba16761SJacob Faibussowitsch PetscErrorCode PetscErrorMessage(PetscErrorCode errnum, const char *text[], char **specific)
257d71ae5a4SJacob Faibussowitsch {
2583ba16761SJacob Faibussowitsch   PetscFunctionBegin;
2593ba16761SJacob Faibussowitsch   if (text) {
2603ba16761SJacob Faibussowitsch     if (errnum > PETSC_ERR_MIN_VALUE && errnum < PETSC_ERR_MAX_VALUE) {
261c35ec7c7SPierre Jolivet       size_t len;
262c35ec7c7SPierre Jolivet 
263c35ec7c7SPierre Jolivet       *text = PetscErrorStrings[errnum - PETSC_ERR_MIN_VALUE - 1];
264c35ec7c7SPierre Jolivet       PetscCall(PetscStrlen(*text, &len));
265c35ec7c7SPierre Jolivet       if (!len) *text = NULL;
2663ba16761SJacob Faibussowitsch     } else if (errnum == PETSC_ERR_BOOLEAN_MACRO_FAILURE) {
2673ba16761SJacob Faibussowitsch       /* this "error code" arises from failures in boolean macros, where the || operator is
2683ba16761SJacob Faibussowitsch          used to short-circuit the macro call in case of error. This has the side effect of
2693ba16761SJacob Faibussowitsch          "returning" either 0 (PETSC_SUCCESS) or 1 (PETSC_ERR_UNKNONWN):
270e5c89e4eSSatish Balay 
2713ba16761SJacob Faibussowitsch          #define PETSC_FOO(x) ((PetscErrorCode)(PetscBar(x) || PetscBaz(x)))
2723ba16761SJacob Faibussowitsch 
2733ba16761SJacob Faibussowitsch          If PetscBar() fails (returns nonzero) PetscBaz() is not executed but the result of
2743ba16761SJacob Faibussowitsch          this expression is boolean false, hence PETSC_ERR_UNNOWN
2753ba16761SJacob Faibussowitsch        */
2763ba16761SJacob Faibussowitsch       *text = "Error occurred in boolean shortcuit in macro";
2773ba16761SJacob Faibussowitsch     } else {
2783ba16761SJacob Faibussowitsch       *text = NULL;
2793ba16761SJacob Faibussowitsch     }
2803ba16761SJacob Faibussowitsch   }
281a297a907SKarl Rupp   if (specific) *specific = PetscErrorBaseMessage;
2823ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
283e5c89e4eSSatish Balay }
284e5c89e4eSSatish Balay 
285984a1229SMatthew G. Knepley #if defined(PETSC_CLANGUAGE_CXX)
286984a1229SMatthew G. Knepley   /* C++ exceptions are formally not allowed to propagate through extern "C" code. In practice, far too much software
287984a1229SMatthew G. Knepley  * would be broken if implementations did not handle it it some common cases. However, keep in mind
288984a1229SMatthew G. Knepley  *
289984a1229SMatthew G. Knepley  *   Rule 62. Don't allow exceptions to propagate across module boundaries
290984a1229SMatthew G. Knepley  *
291984a1229SMatthew G. Knepley  * in "C++ Coding Standards" by Sutter and Alexandrescu. (This accounts for part of the ongoing C++ binary interface
292984a1229SMatthew G. Knepley  * instability.) Having PETSc raise errors as C++ exceptions was probably misguided and should eventually be removed.
293984a1229SMatthew G. Knepley  *
294984a1229SMatthew G. Knepley  * Here is the problem: You have a C++ function call a PETSc function, and you would like to maintain the error message
295984a1229SMatthew G. Knepley  * and stack information from the PETSc error. You could make everyone write exactly this code in their C++, but that
296984a1229SMatthew G. Knepley  * seems crazy to me.
297984a1229SMatthew G. Knepley  */
298984a1229SMatthew G. Knepley   #include <sstream>
2994c94c282SMatthew G. Knepley   #include <stdexcept>
300d71ae5a4SJacob Faibussowitsch static void PetscCxxErrorThrow()
301d71ae5a4SJacob Faibussowitsch {
302984a1229SMatthew G. Knepley   const char *str;
303984a1229SMatthew G. Knepley   if (eh && eh->ctx) {
304984a1229SMatthew G. Knepley     std::ostringstream *msg;
305984a1229SMatthew G. Knepley     msg = (std::ostringstream *)eh->ctx;
306984a1229SMatthew G. Knepley     str = msg->str().c_str();
307984a1229SMatthew G. Knepley   } else str = "Error detected in C PETSc";
308984a1229SMatthew G. Knepley 
309984a1229SMatthew G. Knepley   throw std::runtime_error(str);
310984a1229SMatthew G. Knepley }
311984a1229SMatthew G. Knepley #endif
312984a1229SMatthew G. Knepley 
313e5c89e4eSSatish Balay /*@C
31445b666d6SBarry Smith    PetscError - Routine that is called when an error has been detected, usually called through the macro SETERRQ(PETSC_COMM_SELF,).
315e5c89e4eSSatish Balay 
316811af0c4SBarry Smith   Collective
317e5c89e4eSSatish Balay 
318e5c89e4eSSatish Balay    Input Parameters:
319e32f2f54SBarry Smith +  comm - communicator over which error occurred.  ALL ranks of this communicator MUST call this routine
320e32f2f54SBarry Smith .  line - the line number of the error (indicated by __LINE__)
3213de71b31SHong Zhang .  func - the function name in which the error was detected
322e5c89e4eSSatish Balay .  file - the file in which the error was detected (indicated by __FILE__)
323e5c89e4eSSatish Balay .  n - the generic error number
324811af0c4SBarry Smith .  p - `PETSC_ERROR_INITIAL` indicates the error was initially detected, `PETSC_ERROR_REPEAT` indicates this is a traceback from a previously detected error
325e5c89e4eSSatish Balay -  mess - formatted message string - aka printf
326e5c89e4eSSatish Balay 
327811af0c4SBarry Smith   Options Database Keys:
32845b666d6SBarry Smith +  -error_output_stdout - output the error messages to stdout instead of the default stderr
32945b666d6SBarry Smith -  -error_output_none - do not output the error messages
33045b666d6SBarry Smith 
331e5c89e4eSSatish Balay   Level: intermediate
332e5c89e4eSSatish Balay 
333e5c89e4eSSatish Balay    Notes:
334e809461dSJacob Faibussowitsch    PETSc error handling is done with error return codes. A non-zero return indicates an error
335e809461dSJacob Faibussowitsch    was detected. The return-value of this routine is what is ultimately returned by
336e809461dSJacob Faibussowitsch    `SETERRQ()`.
33745b666d6SBarry Smith 
338e809461dSJacob Faibussowitsch    Note that numerical errors (potential divide by zero, for example) are not managed by the
339e809461dSJacob Faibussowitsch    error return codes; they are managed via, for example, `KSPGetConvergedReason()` that
340e809461dSJacob Faibussowitsch    indicates if the solve was successful or not. The option `-ksp_error_if_not_converged`, for
341e809461dSJacob Faibussowitsch    example, turns numerical failures into hard errors managed via `PetscError()`.
34245b666d6SBarry Smith 
343e809461dSJacob Faibussowitsch    PETSc provides a rich supply of error handlers, see the list below, and users can also
344e809461dSJacob Faibussowitsch    provide their own error handlers.
345811af0c4SBarry Smith 
346e809461dSJacob Faibussowitsch    If the user sets their own error handler (via `PetscPushErrorHandler()`) they may return any
347e809461dSJacob Faibussowitsch    arbitrary value from it, but are encouraged to return nonzero values. If the return value is
348e809461dSJacob Faibussowitsch    zero, `SETERRQ()` will ignore the value and return `PETSC_ERR_RETURN` (a nonzero value)
349e809461dSJacob Faibussowitsch    instead.
350e809461dSJacob Faibussowitsch 
351e809461dSJacob Faibussowitsch    Most users need not directly use this routine and the error handlers, but can instead use
352e809461dSJacob Faibussowitsch    the simplified interface `PetscCall()` or `SETERRQ()`.
353e5c89e4eSSatish Balay 
354e3081792SBarry Smith    Fortran Note:
355e3081792SBarry Smith    This routine is used differently from Fortran
356e3081792SBarry Smith $    PetscError(MPI_Comm comm,PetscErrorCode n,PetscErrorType p,char *message)
357e3081792SBarry Smith 
358811af0c4SBarry Smith    Developer Note:
359811af0c4SBarry Smith    Since this is called after an error condition it should not be calling any error handlers (currently it ignores any error codes)
3603bf036e2SBarry 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
3613bf036e2SBarry Smith    but this annoying.
3623bf036e2SBarry Smith 
363db781477SPatrick Sanan .seealso: `PetscErrorCode`, `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`,
364db781477SPatrick Sanan           `PetscReturnErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscEmacsClientErrorHandler()`,
365db781477SPatrick Sanan           `SETERRQ()`, `PetscCall()`, `CHKMEMQ`, `SETERRQ()`, `SETERRQ()`, `PetscErrorMessage()`, `PETSCABORT()`
366e5c89e4eSSatish Balay @*/
367d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscError(MPI_Comm comm, int line, const char *func, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, ...)
368d71ae5a4SJacob Faibussowitsch {
369e5c89e4eSSatish Balay   va_list        Argp;
370c9a19010SBarry Smith   size_t         fullLength;
37102c9f0b5SLisandro Dalcin   char           buf[2048], *lbuf = NULL;
372fbfcfee5SBarry Smith   PetscBool      ismain;
3733bf036e2SBarry Smith   PetscErrorCode ierr;
374e5c89e4eSSatish Balay 
37539a651e2SJacob Faibussowitsch   if (!PetscErrorHandlingInitialized) return n;
3760d577829SBarry Smith   if (comm == MPI_COMM_NULL) comm = PETSC_COMM_SELF;
377e5c89e4eSSatish Balay 
378e5c89e4eSSatish Balay   /* Compose the message evaluating the print format */
379e5c89e4eSSatish Balay   if (mess) {
380e5c89e4eSSatish Balay     va_start(Argp, mess);
3813ba16761SJacob Faibussowitsch     ierr = PetscVSNPrintf(buf, 2048, mess, &fullLength, Argp);
382e5c89e4eSSatish Balay     va_end(Argp);
383e5c89e4eSSatish Balay     lbuf = buf;
3849b15cf9aSJacob Faibussowitsch     if (p == PETSC_ERROR_INITIAL) ierr = PetscStrncpy(PetscErrorBaseMessage, lbuf, sizeof(PetscErrorBaseMessage));
385e5c89e4eSSatish Balay   }
386e5c89e4eSSatish Balay 
3873ba16761SJacob Faibussowitsch   if (p == PETSC_ERROR_INITIAL && n != PETSC_ERR_MEMC) ierr = PetscMallocValidate(__LINE__, PETSC_FUNCTION_NAME, __FILE__);
3884ed0ab5bSBarry Smith 
38902c9f0b5SLisandro Dalcin   if (!eh) ierr = PetscTraceBackErrorHandler(comm, line, func, file, n, p, lbuf, NULL);
390efca3c55SSatish Balay   else ierr = (*eh->handler)(comm, line, func, file, n, p, lbuf, eh->ctx);
391362febeeSStefano Zampini   PetscStackClearTop;
392e5c89e4eSSatish Balay 
393e5c89e4eSSatish Balay   /*
3947233276eSBarry Smith       If this is called from the main() routine we call MPI_Abort() instead of
395e5c89e4eSSatish Balay     return to allow the parallel program to be properly shutdown.
396e5c89e4eSSatish Balay 
3977233276eSBarry Smith     Does not call PETSCABORT() since that would provide the wrong source file and line number information
398e5c89e4eSSatish Balay   */
39949c86fc7SBarry Smith   if (func) {
4003ba16761SJacob Faibussowitsch     PetscErrorCode cmp_ierr = PetscStrncmp(func, "main", 4, &ismain);
4017233276eSBarry Smith     if (ismain) {
4023ba16761SJacob Faibussowitsch       if (petscwaitonerrorflg) cmp_ierr = PetscSleep(1000);
4033ba16761SJacob Faibussowitsch       (void)cmp_ierr;
404660278c0SBarry Smith       PETSCABORT(comm, ierr);
4057233276eSBarry Smith     }
40649c86fc7SBarry Smith   }
4072c280183SJed Brown #if defined(PETSC_CLANGUAGE_CXX)
408ad540459SPierre Jolivet   if (p == PETSC_ERROR_IN_CXX) PetscCxxErrorThrow();
409d736bfebSBarry Smith #endif
410362febeeSStefano Zampini   return ierr;
411e5c89e4eSSatish Balay }
412e5c89e4eSSatish Balay 
413e5c89e4eSSatish Balay /* -------------------------------------------------------------------------*/
414e5c89e4eSSatish Balay 
415e5c89e4eSSatish Balay /*@C
416e5c89e4eSSatish Balay     PetscIntView - Prints an array of integers; useful for debugging.
417e5c89e4eSSatish Balay 
418c3339decSBarry Smith     Collective
419e5c89e4eSSatish Balay 
420e5c89e4eSSatish Balay     Input Parameters:
421e5c89e4eSSatish Balay +   N - number of integers in array
422e5c89e4eSSatish Balay .   idx - array of integers
423811af0c4SBarry Smith -   viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
424e5c89e4eSSatish Balay 
425e5c89e4eSSatish Balay   Level: intermediate
426e5c89e4eSSatish Balay 
427811af0c4SBarry Smith     Note:
428811af0c4SBarry Smith     This may be called from within the debugger
429300a7f5bSBarry Smith 
430811af0c4SBarry Smith     Developer Note:
431811af0c4SBarry Smith     idx cannot be const because may be passed to binary viewer where temporary byte swapping may be done
432811af0c4SBarry Smith 
433811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscRealView()`
434e5c89e4eSSatish Balay @*/
435d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscIntView(PetscInt N, const PetscInt idx[], PetscViewer viewer)
436d71ae5a4SJacob Faibussowitsch {
437ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
438e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 20, p = N % 20;
439ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
440e5c89e4eSSatish Balay   MPI_Comm    comm;
441e5c89e4eSSatish Balay 
442e5c89e4eSSatish Balay   PetscFunctionBegin;
443e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
4440700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
4455ab33896SBarry Smith   if (N) PetscValidIntPointer(idx, 2);
4469566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
4479566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
4489566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
449e5c89e4eSSatish Balay 
4509566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
4519566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
452e5c89e4eSSatish Balay   if (iascii) {
4539566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
454e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
455ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4569566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * i));
457ca0c3be5SJacob Faibussowitsch       } else {
4589566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * i));
459ca0c3be5SJacob Faibussowitsch       }
46048a46eb9SPierre Jolivet       for (j = 0; j < 20; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[i * 20 + j]));
4619566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
462e5c89e4eSSatish Balay     }
463e5c89e4eSSatish Balay     if (p) {
464ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4659566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * n));
466ca0c3be5SJacob Faibussowitsch       } else {
4679566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * n));
468ca0c3be5SJacob Faibussowitsch       }
4699566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[20 * n + i]));
4709566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
471e5c89e4eSSatish Balay     }
4729566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
4739566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
4746805f65bSBarry Smith   } else if (isbinary) {
475ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
476e5c89e4eSSatish Balay     PetscInt    *array;
477783b601eSJed Brown 
4789566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
479e5c89e4eSSatish Balay 
480e5c89e4eSSatish Balay     if (size > 1) {
481e5c89e4eSSatish Balay       if (rank) {
4829566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
4839566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, NULL, NULL, NULL, MPIU_INT, 0, comm));
484e5c89e4eSSatish Balay       } else {
4859566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
4869566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
487e5c89e4eSSatish Balay         Ntotal = sizes[0];
4889566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
489e5c89e4eSSatish Balay         displs[0] = 0;
490e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
491e5c89e4eSSatish Balay           Ntotal += sizes[i];
492e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
493e5c89e4eSSatish Balay         }
4949566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
4959566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, array, sizes, displs, MPIU_INT, 0, comm));
4969566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_INT));
4979566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
4989566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
4999566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
500e5c89e4eSSatish Balay       }
501e5c89e4eSSatish Balay     } else {
5029566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, idx, N, PETSC_INT));
503e5c89e4eSSatish Balay     }
504e5c89e4eSSatish Balay   } else {
505e5c89e4eSSatish Balay     const char *tname;
5069566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
50798921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
508e5c89e4eSSatish Balay   }
5093ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
510e5c89e4eSSatish Balay }
511e5c89e4eSSatish Balay 
512e5c89e4eSSatish Balay /*@C
513e5c89e4eSSatish Balay     PetscRealView - Prints an array of doubles; useful for debugging.
514e5c89e4eSSatish Balay 
515c3339decSBarry Smith     Collective
516e5c89e4eSSatish Balay 
517e5c89e4eSSatish Balay     Input Parameters:
518811af0c4SBarry Smith +   N - number of `PetscReal` in array
519811af0c4SBarry Smith .   idx - array of `PetscReal`
520811af0c4SBarry Smith -   viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
521e5c89e4eSSatish Balay 
522e5c89e4eSSatish Balay   Level: intermediate
523e5c89e4eSSatish Balay 
524811af0c4SBarry Smith     Note:
525811af0c4SBarry Smith     This may be called from within the debugger
526300a7f5bSBarry Smith 
527811af0c4SBarry Smith     Developer Note:
528811af0c4SBarry Smith     idx cannot be const because may be passed to binary viewer where temporary byte swapping may be done
529811af0c4SBarry Smith 
530811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`
531e5c89e4eSSatish Balay @*/
532d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscRealView(PetscInt N, const PetscReal idx[], PetscViewer viewer)
533d71ae5a4SJacob Faibussowitsch {
534ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
535e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 5, p = N % 5;
536ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
537e5c89e4eSSatish Balay   MPI_Comm    comm;
538e5c89e4eSSatish Balay 
539e5c89e4eSSatish Balay   PetscFunctionBegin;
540e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
5410700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
542064a246eSJacob Faibussowitsch   PetscValidRealPointer(idx, 2);
5439566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
5449566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
5459566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
546e5c89e4eSSatish Balay 
5479566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
5489566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
549e5c89e4eSSatish Balay   if (iascii) {
5501a989b97SToby Isaac     PetscInt tab;
5511a989b97SToby Isaac 
5529566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
5539566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIGetTab(viewer, &tab));
554e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
5559566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
556ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5579566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * i));
558ca0c3be5SJacob Faibussowitsch       } else {
5599566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * i));
560ca0c3be5SJacob Faibussowitsch       }
5619566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
56248a46eb9SPierre Jolivet       for (j = 0; j < 5; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 5 + j]));
5639566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
564e5c89e4eSSatish Balay     }
565e5c89e4eSSatish Balay     if (p) {
5669566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
567ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5689566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * n));
569ca0c3be5SJacob Faibussowitsch       } else {
5709566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * n));
571ca0c3be5SJacob Faibussowitsch       }
5729566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
5739566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[5 * n + i]));
5749566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
575e5c89e4eSSatish Balay     }
5769566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
5779566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIISetTab(viewer, tab));
5789566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
5796805f65bSBarry Smith   } else if (isbinary) {
580ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, *displs, Ntotal, NN;
581e5c89e4eSSatish Balay     PetscReal   *array;
582e5c89e4eSSatish Balay 
5839566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
584e5c89e4eSSatish Balay 
585e5c89e4eSSatish Balay     if (size > 1) {
586e5c89e4eSSatish Balay       if (rank) {
5879566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
5889566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, NULL, NULL, NULL, MPIU_REAL, 0, comm));
589e5c89e4eSSatish Balay       } else {
5909566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
5919566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
592e5c89e4eSSatish Balay         Ntotal = sizes[0];
5939566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
594e5c89e4eSSatish Balay         displs[0] = 0;
595e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
596e5c89e4eSSatish Balay           Ntotal += sizes[i];
597e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
598e5c89e4eSSatish Balay         }
5999566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
6009566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, array, sizes, displs, MPIU_REAL, 0, comm));
6019566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_REAL));
6029566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
6039566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
6049566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
605e5c89e4eSSatish Balay       }
606e5c89e4eSSatish Balay     } else {
6079566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_REAL));
608e5c89e4eSSatish Balay     }
609e5c89e4eSSatish Balay   } else {
610e5c89e4eSSatish Balay     const char *tname;
6119566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
61298921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
613e5c89e4eSSatish Balay   }
6143ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
615e5c89e4eSSatish Balay }
616e5c89e4eSSatish Balay 
617e5c89e4eSSatish Balay /*@C
618811af0c4SBarry Smith     PetscScalarView - Prints an array of `PetscScalar`; useful for debugging.
619e5c89e4eSSatish Balay 
620c3339decSBarry Smith     Collective
621e5c89e4eSSatish Balay 
622e5c89e4eSSatish Balay     Input Parameters:
623e5c89e4eSSatish Balay +   N - number of scalars in array
624e5c89e4eSSatish Balay .   idx - array of scalars
625811af0c4SBarry Smith -   viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
626e5c89e4eSSatish Balay 
627e5c89e4eSSatish Balay   Level: intermediate
628e5c89e4eSSatish Balay 
629811af0c4SBarry Smith     Note:
630811af0c4SBarry Smith     This may be called from within the debugger
631300a7f5bSBarry Smith 
632811af0c4SBarry Smith     Developer Note:
633811af0c4SBarry Smith     idx cannot be const because may be passed to binary viewer where byte swapping may be done
634811af0c4SBarry Smith 
635811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`, `PetscRealView()`
636e5c89e4eSSatish Balay @*/
637d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscScalarView(PetscInt N, const PetscScalar idx[], PetscViewer viewer)
638d71ae5a4SJacob Faibussowitsch {
639ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
640e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 3, p = N % 3;
641ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
642e5c89e4eSSatish Balay   MPI_Comm    comm;
643e5c89e4eSSatish Balay 
644e5c89e4eSSatish Balay   PetscFunctionBegin;
645e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
646e5c89e4eSSatish Balay   PetscValidHeader(viewer, 3);
6478c34849bSStefano Zampini   if (N) PetscValidScalarPointer(idx, 2);
6489566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
6499566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
6509566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
651e5c89e4eSSatish Balay 
6529566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
6539566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
654e5c89e4eSSatish Balay   if (iascii) {
6559566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
656e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
657ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6589566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * i));
659ca0c3be5SJacob Faibussowitsch       } else {
6609566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * i));
661ca0c3be5SJacob Faibussowitsch       }
662e5c89e4eSSatish Balay       for (j = 0; j < 3; j++) {
663e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6649566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[i * 3 + j]), (double)PetscImaginaryPart(idx[i * 3 + j])));
665e5c89e4eSSatish Balay #else
6669566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 3 + j]));
667e5c89e4eSSatish Balay #endif
668e5c89e4eSSatish Balay       }
6699566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
670e5c89e4eSSatish Balay     }
671e5c89e4eSSatish Balay     if (p) {
672ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6739566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * n));
674ca0c3be5SJacob Faibussowitsch       } else {
6759566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * n));
676ca0c3be5SJacob Faibussowitsch       }
677e5c89e4eSSatish Balay       for (i = 0; i < p; i++) {
678e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6799566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[n * 3 + i]), (double)PetscImaginaryPart(idx[n * 3 + i])));
680e5c89e4eSSatish Balay #else
6819566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[3 * n + i]));
682e5c89e4eSSatish Balay #endif
683e5c89e4eSSatish Balay       }
6849566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
685e5c89e4eSSatish Balay     }
6869566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
6879566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
6886805f65bSBarry Smith   } else if (isbinary) {
689ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
690e5c89e4eSSatish Balay     PetscScalar *array;
691e5c89e4eSSatish Balay 
6929566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
693e5c89e4eSSatish Balay 
694e5c89e4eSSatish Balay     if (size > 1) {
695e5c89e4eSSatish Balay       if (rank) {
6969566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
6979566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, NULL, NULL, NULL, MPIU_SCALAR, 0, comm));
698e5c89e4eSSatish Balay       } else {
6999566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
7009566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
701e5c89e4eSSatish Balay         Ntotal = sizes[0];
7029566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
703e5c89e4eSSatish Balay         displs[0] = 0;
704e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
705e5c89e4eSSatish Balay           Ntotal += sizes[i];
706e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
707e5c89e4eSSatish Balay         }
7089566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
7099566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, array, sizes, displs, MPIU_SCALAR, 0, comm));
7109566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_SCALAR));
7119566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
7129566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
7139566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
714e5c89e4eSSatish Balay       }
715e5c89e4eSSatish Balay     } else {
7169566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_SCALAR));
717e5c89e4eSSatish Balay     }
718e5c89e4eSSatish Balay   } else {
719e5c89e4eSSatish Balay     const char *tname;
7209566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
72198921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
722e5c89e4eSSatish Balay   }
7233ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
724e5c89e4eSSatish Balay }
725e5c89e4eSSatish Balay 
726e22e20c5SJunchao Zhang #if defined(PETSC_HAVE_CUDA)
7270e6b6b59SJacob Faibussowitsch   #include <petscdevice_cuda.h>
728d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUBLASGetErrorName(cublasStatus_t status)
729d71ae5a4SJacob Faibussowitsch {
730e22e20c5SJunchao Zhang   switch (status) {
731e22e20c5SJunchao Zhang   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
732d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_SUCCESS:
733d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_SUCCESS";
734d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_INITIALIZED:
735d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_INITIALIZED";
736d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ALLOC_FAILED:
737d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ALLOC_FAILED";
738d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INVALID_VALUE:
739d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INVALID_VALUE";
740d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ARCH_MISMATCH:
741d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ARCH_MISMATCH";
742d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_MAPPING_ERROR:
743d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_MAPPING_ERROR";
744d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_EXECUTION_FAILED:
745d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_EXECUTION_FAILED";
746d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INTERNAL_ERROR:
747d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INTERNAL_ERROR";
748d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_SUPPORTED:
749d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_SUPPORTED";
750d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_LICENSE_ERROR:
751d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_LICENSE_ERROR";
752e22e20c5SJunchao Zhang   #endif
753d71ae5a4SJacob Faibussowitsch   default:
754d71ae5a4SJacob Faibussowitsch     return "unknown error";
755e22e20c5SJunchao Zhang   }
756e22e20c5SJunchao Zhang }
757d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUSolverGetErrorName(cusolverStatus_t status)
758d71ae5a4SJacob Faibussowitsch {
759a4b895e1SBarry Smith   switch (status) {
760a4b895e1SBarry Smith   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
761d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_SUCCESS:
762d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_SUCCESS";
763d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_INITIALIZED:
764d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_INITIALIZED";
765d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_VALUE:
766d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_VALUE";
767d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ARCH_MISMATCH:
768d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ARCH_MISMATCH";
769d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INTERNAL_ERROR:
770d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INTERNAL_ERROR";
771030f984aSJacob Faibussowitsch     #if (CUDART_VERSION >= 9000) /* CUDA 9.0 had these defined on June 2021 */
772d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ALLOC_FAILED:
773d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ALLOC_FAILED";
774d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MAPPING_ERROR:
775d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MAPPING_ERROR";
776d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_EXECUTION_FAILED:
777d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_EXECUTION_FAILED";
778d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
779d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
780d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_SUPPORTED:
781d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_SUPPORTED ";
782d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ZERO_PIVOT:
783d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ZERO_PIVOT";
784d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_LICENSE:
785d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_LICENSE";
786a4b895e1SBarry Smith     #endif
787030f984aSJacob Faibussowitsch   #endif
788d71ae5a4SJacob Faibussowitsch   default:
789d71ae5a4SJacob Faibussowitsch     return "unknown error";
790030f984aSJacob Faibussowitsch   }
791030f984aSJacob Faibussowitsch }
792d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUFFTGetErrorName(cufftResult result)
793d71ae5a4SJacob Faibussowitsch {
794030f984aSJacob Faibussowitsch   switch (result) {
795d71ae5a4SJacob Faibussowitsch   case CUFFT_SUCCESS:
796d71ae5a4SJacob Faibussowitsch     return "CUFFT_SUCCESS";
797d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_PLAN:
798d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_PLAN";
799d71ae5a4SJacob Faibussowitsch   case CUFFT_ALLOC_FAILED:
800d71ae5a4SJacob Faibussowitsch     return "CUFFT_ALLOC_FAILED";
801d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_TYPE:
802d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_TYPE";
803d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_VALUE:
804d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_VALUE";
805d71ae5a4SJacob Faibussowitsch   case CUFFT_INTERNAL_ERROR:
806d71ae5a4SJacob Faibussowitsch     return "CUFFT_INTERNAL_ERROR";
807d71ae5a4SJacob Faibussowitsch   case CUFFT_EXEC_FAILED:
808d71ae5a4SJacob Faibussowitsch     return "CUFFT_EXEC_FAILED";
809d71ae5a4SJacob Faibussowitsch   case CUFFT_SETUP_FAILED:
810d71ae5a4SJacob Faibussowitsch     return "CUFFT_SETUP_FAILED";
811d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_SIZE:
812d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_SIZE";
813d71ae5a4SJacob Faibussowitsch   case CUFFT_UNALIGNED_DATA:
814d71ae5a4SJacob Faibussowitsch     return "CUFFT_UNALIGNED_DATA";
815d71ae5a4SJacob Faibussowitsch   case CUFFT_INCOMPLETE_PARAMETER_LIST:
816d71ae5a4SJacob Faibussowitsch     return "CUFFT_INCOMPLETE_PARAMETER_LIST";
817d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_DEVICE:
818d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_DEVICE";
819d71ae5a4SJacob Faibussowitsch   case CUFFT_PARSE_ERROR:
820d71ae5a4SJacob Faibussowitsch     return "CUFFT_PARSE_ERROR";
821d71ae5a4SJacob Faibussowitsch   case CUFFT_NO_WORKSPACE:
822d71ae5a4SJacob Faibussowitsch     return "CUFFT_NO_WORKSPACE";
823d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_IMPLEMENTED:
824d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_IMPLEMENTED";
825d71ae5a4SJacob Faibussowitsch   case CUFFT_LICENSE_ERROR:
826d71ae5a4SJacob Faibussowitsch     return "CUFFT_LICENSE_ERROR";
827d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_SUPPORTED:
828d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_SUPPORTED";
829d71ae5a4SJacob Faibussowitsch   default:
830d71ae5a4SJacob Faibussowitsch     return "unknown error";
831a4b895e1SBarry Smith   }
832a4b895e1SBarry Smith }
833e22e20c5SJunchao Zhang #endif
83459af0bd3SScott Kruger 
83559af0bd3SScott Kruger #if defined(PETSC_HAVE_HIP)
8360e6b6b59SJacob Faibussowitsch   #include <petscdevice_hip.h>
837d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscHIPBLASGetErrorName(hipblasStatus_t status)
838d71ae5a4SJacob Faibussowitsch {
83959af0bd3SScott Kruger   switch (status) {
840d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_SUCCESS:
841d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_SUCCESS";
842d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_INITIALIZED:
843d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_INITIALIZED";
844d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ALLOC_FAILED:
845d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ALLOC_FAILED";
846d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INVALID_VALUE:
847d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INVALID_VALUE";
848d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ARCH_MISMATCH:
849d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ARCH_MISMATCH";
850d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_MAPPING_ERROR:
851d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_MAPPING_ERROR";
852d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_EXECUTION_FAILED:
853d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_EXECUTION_FAILED";
854d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INTERNAL_ERROR:
855d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INTERNAL_ERROR";
856d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_SUPPORTED:
857d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_SUPPORTED";
858d71ae5a4SJacob Faibussowitsch   default:
859d71ae5a4SJacob Faibussowitsch     return "unknown error";
86059af0bd3SScott Kruger   }
86159af0bd3SScott Kruger }
86247d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSPARSEGetErrorName(hipsparseStatus_t status)
86347d993e7Ssuyashtn {
86447d993e7Ssuyashtn   switch (status) {
86547d993e7Ssuyashtn   case HIPSPARSE_STATUS_SUCCESS:
86647d993e7Ssuyashtn     return "HIPSPARSE_STATUS_SUCCESS";
86747d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_INITIALIZED:
86847d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_INITIALIZED";
86947d993e7Ssuyashtn   case HIPSPARSE_STATUS_ALLOC_FAILED:
87047d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ALLOC_FAILED";
87147d993e7Ssuyashtn   case HIPSPARSE_STATUS_INVALID_VALUE:
87247d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INVALID_VALUE";
87347d993e7Ssuyashtn   case HIPSPARSE_STATUS_ARCH_MISMATCH:
87447d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ARCH_MISMATCH";
87547d993e7Ssuyashtn   case HIPSPARSE_STATUS_MAPPING_ERROR:
87647d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MAPPING_ERROR";
87747d993e7Ssuyashtn   case HIPSPARSE_STATUS_EXECUTION_FAILED:
87847d993e7Ssuyashtn     return "HIPSPARSE_STATUS_EXECUTION_FAILED";
87947d993e7Ssuyashtn   case HIPSPARSE_STATUS_INTERNAL_ERROR:
88047d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INTERNAL_ERROR";
88147d993e7Ssuyashtn   case HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
88247d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
88347d993e7Ssuyashtn   case HIPSPARSE_STATUS_ZERO_PIVOT:
88447d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ZERO_PIVOT";
88547d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_SUPPORTED:
88647d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_SUPPORTED";
88747d993e7Ssuyashtn   case HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES:
88847d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES";
88947d993e7Ssuyashtn   default:
89047d993e7Ssuyashtn     return "unknown error";
89147d993e7Ssuyashtn   }
89247d993e7Ssuyashtn }
89347d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSolverGetErrorName(hipsolverStatus_t status)
89447d993e7Ssuyashtn {
89547d993e7Ssuyashtn   switch (status) {
89647d993e7Ssuyashtn   case HIPSOLVER_STATUS_SUCCESS:
89747d993e7Ssuyashtn     return "HIPSOLVER_STATUS_SUCCESS";
89847d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_INITIALIZED:
89947d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_INITIALIZED";
90047d993e7Ssuyashtn   case HIPSOLVER_STATUS_ALLOC_FAILED:
90147d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ALLOC_FAILED";
90247d993e7Ssuyashtn   case HIPSOLVER_STATUS_MAPPING_ERROR:
90347d993e7Ssuyashtn     return "HIPSOLVER_STATUS_MAPPING_ERROR";
90447d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_VALUE:
90547d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_VALUE";
90647d993e7Ssuyashtn   case HIPSOLVER_STATUS_EXECUTION_FAILED:
90747d993e7Ssuyashtn     return "HIPSOLVER_STATUS_EXECUTION_FAILED";
90847d993e7Ssuyashtn   case HIPSOLVER_STATUS_INTERNAL_ERROR:
90947d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INTERNAL_ERROR";
91047d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_SUPPORTED:
91147d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_SUPPORTED ";
91247d993e7Ssuyashtn   case HIPSOLVER_STATUS_ARCH_MISMATCH:
91347d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ARCH_MISMATCH";
91447d993e7Ssuyashtn   case HIPSOLVER_STATUS_HANDLE_IS_NULLPTR:
91547d993e7Ssuyashtn     return "HIPSOLVER_STATUS_HANDLE_IS_NULLPTR";
91647d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_ENUM:
91747d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_ENUM";
91847d993e7Ssuyashtn   case HIPSOLVER_STATUS_UNKNOWN:
91947d993e7Ssuyashtn   default:
92047d993e7Ssuyashtn     return "HIPSOLVER_STATUS_UNKNOWN";
92147d993e7Ssuyashtn   }
92247d993e7Ssuyashtn }
92359af0bd3SScott Kruger #endif
924db9cea48SBarry Smith 
925db9cea48SBarry Smith /*@
926811af0c4SBarry Smith       PetscMPIErrorString - Given an MPI error code returns the `MPI_Error_string()` appropriately
927db9cea48SBarry Smith            formatted for displaying with the PETSc error handlers.
928db9cea48SBarry Smith 
929db9cea48SBarry Smith  Input Parameter:
930db9cea48SBarry Smith .  err - the MPI error code
931db9cea48SBarry Smith 
932db9cea48SBarry Smith  Output Parameter:
933811af0c4SBarry Smith .  string - the MPI error message, should declare its length to be larger than `MPI_MAX_ERROR_STRING`
934db9cea48SBarry Smith 
93549c86fc7SBarry Smith    Level: developer
93649c86fc7SBarry Smith 
937811af0c4SBarry Smith  Note:
938db9cea48SBarry Smith     Does not return an error code or do error handling because it may be called from inside an error handler
939db9cea48SBarry Smith 
940db9cea48SBarry Smith @*/
941d71ae5a4SJacob Faibussowitsch void PetscMPIErrorString(PetscMPIInt err, char *string)
942d71ae5a4SJacob Faibussowitsch {
943db9cea48SBarry Smith   char        errorstring[MPI_MAX_ERROR_STRING];
944db9cea48SBarry Smith   PetscMPIInt len, j = 0;
945db9cea48SBarry Smith 
946db9cea48SBarry Smith   MPI_Error_string(err, (char *)errorstring, &len);
947db9cea48SBarry Smith   for (PetscMPIInt i = 0; i < len; i++) {
948db9cea48SBarry Smith     string[j++] = errorstring[i];
949db9cea48SBarry Smith     if (errorstring[i] == '\n') {
950db9cea48SBarry Smith       for (PetscMPIInt k = 0; k < 16; k++) string[j++] = ' ';
951db9cea48SBarry Smith     }
952db9cea48SBarry Smith   }
953db9cea48SBarry Smith   string[j] = 0;
954db9cea48SBarry Smith }
955