xref: /petsc/src/sys/error/err.c (revision 3ba1676111f5c958fe6c2729b46ca4d523958bb3)
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   }
77*3ba16761SJacob 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 
90e5c89e4eSSatish Balay    Calling sequence of handler:
91efca3c55SSatish Balay $    int handler(MPI_Comm comm,int line,char *func,char *file,PetscErrorCode n,int p,char *mess,void *ctx);
92e5c89e4eSSatish Balay 
93a5b23f4aSJose E. Roman +  comm - communicator over which error occurred
94e5c89e4eSSatish Balay .  line - the line number of the error (indicated by __LINE__)
95e5c89e4eSSatish Balay .  file - the file in which the error was detected (indicated by __FILE__)
96e5c89e4eSSatish Balay .  n - the generic error number (see list defined in include/petscerror.h)
97668f157eSBarry 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 
111811af0c4SBarry Smith    Fortran Note:
11295452b02SPatrick Sanan     You can only push one error handler from Fortran before poping it.
1137850c7c0SBarry Smith 
114db781477SPatrick Sanan .seealso: `PetscPopErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscAbortErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscPushSignalHandler()`
115e5c89e4eSSatish Balay @*/
116d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscPushErrorHandler(PetscErrorCode (*handler)(MPI_Comm comm, int, const char *, const char *, PetscErrorCode, PetscErrorType, const char *, void *), 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;
127*3ba16761SJacob 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;
145*3ba16761SJacob Faibussowitsch   if (!eh) PetscFunctionReturn(PETSC_SUCCESS);
146e5c89e4eSSatish Balay   tmp = eh;
147e5c89e4eSSatish Balay   eh  = eh->previous;
1489566063dSJacob Faibussowitsch   PetscCall(PetscFree(tmp));
149*3ba16761SJacob 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__)
160e93bc3c1Svictor .  file - the file in which the error was detected (indicated by __FILE__)
161e93bc3c1Svictor .  mess - an error text string, usually just printed to the screen
162e93bc3c1Svictor .  n - the generic error number
163e93bc3c1Svictor .  p - specific error number
164e93bc3c1Svictor -  ctx - error handler context
165e93bc3c1Svictor 
166e93bc3c1Svictor    Level: developer
167e93bc3c1Svictor 
168e93bc3c1Svictor    Notes:
169e93bc3c1Svictor    Most users need not directly employ this routine and the other error
170811af0c4SBarry Smith    handlers, but can instead use the simplified interface `SETERRQ()`, which has
171e93bc3c1Svictor    the calling sequence
172e32f2f54SBarry Smith $     SETERRQ(comm,number,mess)
173e93bc3c1Svictor 
174811af0c4SBarry Smith    `PetscIgnoreErrorHandler()` does the same thing as this function, but is deprecated, you should use this function.
175e93bc3c1Svictor 
176811af0c4SBarry Smith    Use `PetscPushErrorHandler()` to set the desired error handler.
177e93bc3c1Svictor 
178db781477SPatrick Sanan .seealso: `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscError()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`, `PetscTraceBackErrorHandler()`,
179db781477SPatrick Sanan           `PetscAttachDebuggerErrorHandler()`, `PetscEmacsClientErrorHandler()`
180e93bc3c1Svictor  @*/
181d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscReturnErrorHandler(MPI_Comm comm, int line, const char *fun, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, void *ctx)
182d71ae5a4SJacob Faibussowitsch {
183362febeeSStefano Zampini   return n;
184e93bc3c1Svictor }
185e93bc3c1Svictor 
186e5c89e4eSSatish Balay static char PetscErrorBaseMessage[1024];
187e5c89e4eSSatish Balay /*
188e5c89e4eSSatish Balay        The numerical values for these are defined in include/petscerror.h; any changes
189e5c89e4eSSatish Balay    there must also be made here
190e5c89e4eSSatish Balay */
191e5c89e4eSSatish Balay static const char *PetscErrorStrings[] = {
192e5c89e4eSSatish Balay   /*55 */ "Out of memory",
193e5c89e4eSSatish Balay   "No support for this operation for this object type",
194e5c89e4eSSatish Balay   "No support for this operation on this system",
195e5c89e4eSSatish Balay   /*58 */ "Operation done in wrong order",
196e5c89e4eSSatish Balay   /*59 */ "Signal received",
197e5c89e4eSSatish Balay   /*60 */ "Nonconforming object sizes",
198e5c89e4eSSatish Balay   "Argument aliasing not permitted",
199e5c89e4eSSatish Balay   "Invalid argument",
200e5c89e4eSSatish Balay   /*63 */ "Argument out of range",
201a17b96a8SKyle Gerard Felker   "Corrupt argument: https://petsc.org/release/faq/#valgrind",
202e5c89e4eSSatish Balay   "Unable to open file",
203e5c89e4eSSatish Balay   "Read from file failed",
204e5c89e4eSSatish Balay   "Write to file failed",
205e5c89e4eSSatish Balay   "Invalid pointer",
206e5c89e4eSSatish Balay   /*69 */ "Arguments must have same type",
207a8b45ee7SBarry Smith   /*70 */ "Attempt to use a pointer that does not point to a valid accessible location",
208a17b96a8SKyle Gerard Felker   /*71 */ "Zero pivot in LU factorization: https://petsc.org/release/faq/#zeropivot",
209e5c89e4eSSatish Balay   /*72 */ "Floating point exception",
210e5c89e4eSSatish Balay   /*73 */ "Object is in wrong state",
211e5c89e4eSSatish Balay   "Corrupted Petsc object",
212e5c89e4eSSatish Balay   "Arguments are incompatible",
213e5c89e4eSSatish Balay   "Error in external library",
214e5c89e4eSSatish Balay   /*77 */ "Petsc has generated inconsistent data",
215a17b96a8SKyle Gerard Felker   "Memory corruption: https://petsc.org/release/faq/#valgrind",
216e5c89e4eSSatish Balay   "Unexpected data in file",
217e5c89e4eSSatish Balay   /*80 */ "Arguments must have same communicators",
218a17b96a8SKyle Gerard Felker   /*81 */ "Zero pivot in Cholesky factorization: https://petsc.org/release/faq/#zeropivot",
219e5c89e4eSSatish Balay   "",
220e5c89e4eSSatish Balay   "",
221a17b96a8SKyle Gerard Felker   "Overflow in integer operation: https://petsc.org/release/faq/#64-bit-indices",
222e5c89e4eSSatish Balay   /*85 */ "Null argument, when expecting valid pointer",
223a17b96a8SKyle Gerard Felker   /*86 */ "Unknown type. Check for miss-spelling or missing package: https://petsc.org/release/install/install/#external-packages",
2243d96e996SBarry Smith   /*87 */ "MPI library at runtime is not compatible with MPI used at compile time",
2258cda6cd7SBarry Smith   /*88 */ "Error in system call",
226a17b96a8SKyle Gerard Felker   /*89 */ "Object Type not set: https://petsc.org/release/faq/#object-type-not-set",
22773260a9bSLisandro Dalcin   /*90 */ "",
22873260a9bSLisandro Dalcin   /*   */ "",
229a17b96a8SKyle Gerard Felker   /*92 */ "See https://petsc.org/release/overview/linear_solve_table/ for possible LU and Cholesky solvers",
230f560318cSPatrick Sanan   /*93 */ "You cannot overwrite this option since that will conflict with other previously set options",
231691b26d3SBarry Smith   /*94 */ "Example/application run with number of MPI ranks it does not support",
232691b26d3SBarry Smith   /*95 */ "Missing or incorrect user input",
233e57d7714SBarry Smith   /*96 */ "GPU resources unavailable",
234764761abSStefano Zampini   /*97 */ "GPU error",
235e809461dSJacob Faibussowitsch   /*98 */ "General MPI error",
236e809461dSJacob Faibussowitsch   /*99 */ "PetscError() incorrectly returned an error code of 0"};
237e5c89e4eSSatish Balay 
238e5c89e4eSSatish Balay /*@C
239*3ba16761SJacob Faibussowitsch   PetscErrorMessage - Returns the text string associated with a PETSc error code.
240e5c89e4eSSatish Balay 
241e5c89e4eSSatish Balay   Not Collective
242e5c89e4eSSatish Balay 
243e5c89e4eSSatish Balay   Input Parameter:
244e5c89e4eSSatish Balay . errnum - the error code
245e5c89e4eSSatish Balay 
246d8d19677SJose E. Roman   Output Parameters:
247*3ba16761SJacob Faibussowitsch + text - the error message (`NULL` if not desired)
248*3ba16761SJacob Faibussowitsch - specific - the specific error message that was set with `SETERRQ()` or
249*3ba16761SJacob Faibussowitsch              `PetscError()`. (`NULL` if not desired)
250e5c89e4eSSatish Balay 
251e5c89e4eSSatish Balay   Level: developer
252e5c89e4eSSatish Balay 
253*3ba16761SJacob Faibussowitsch .seealso: `PetscErrorCode`, `PetscPushErrorHandler()`, `PetscAttachDebuggerErrorHandler()`,
254*3ba16761SJacob Faibussowitsch `PetscError()`, `SETERRQ()`, `PetscCall()` `PetscAbortErrorHandler()`,
255*3ba16761SJacob Faibussowitsch `PetscTraceBackErrorHandler()`
256e5c89e4eSSatish Balay @*/
257*3ba16761SJacob Faibussowitsch PetscErrorCode PetscErrorMessage(PetscErrorCode errnum, const char *text[], char **specific)
258d71ae5a4SJacob Faibussowitsch {
259*3ba16761SJacob Faibussowitsch   PetscFunctionBegin;
260*3ba16761SJacob Faibussowitsch   if (text) {
261*3ba16761SJacob Faibussowitsch     if (errnum > PETSC_ERR_MIN_VALUE && errnum < PETSC_ERR_MAX_VALUE) {
262c35ec7c7SPierre Jolivet       size_t len;
263c35ec7c7SPierre Jolivet 
264c35ec7c7SPierre Jolivet       *text = PetscErrorStrings[errnum - PETSC_ERR_MIN_VALUE - 1];
265c35ec7c7SPierre Jolivet       PetscCall(PetscStrlen(*text, &len));
266c35ec7c7SPierre Jolivet       if (!len) *text = NULL;
267*3ba16761SJacob Faibussowitsch     } else if (errnum == PETSC_ERR_BOOLEAN_MACRO_FAILURE) {
268*3ba16761SJacob Faibussowitsch       /* this "error code" arises from failures in boolean macros, where the || operator is
269*3ba16761SJacob Faibussowitsch          used to short-circuit the macro call in case of error. This has the side effect of
270*3ba16761SJacob Faibussowitsch          "returning" either 0 (PETSC_SUCCESS) or 1 (PETSC_ERR_UNKNONWN):
271e5c89e4eSSatish Balay 
272*3ba16761SJacob Faibussowitsch          #define PETSC_FOO(x) ((PetscErrorCode)(PetscBar(x) || PetscBaz(x)))
273*3ba16761SJacob Faibussowitsch 
274*3ba16761SJacob Faibussowitsch          If PetscBar() fails (returns nonzero) PetscBaz() is not executed but the result of
275*3ba16761SJacob Faibussowitsch          this expression is boolean false, hence PETSC_ERR_UNNOWN
276*3ba16761SJacob Faibussowitsch        */
277*3ba16761SJacob Faibussowitsch       *text = "Error occurred in boolean shortcuit in macro";
278*3ba16761SJacob Faibussowitsch     } else {
279*3ba16761SJacob Faibussowitsch       *text = NULL;
280*3ba16761SJacob Faibussowitsch     }
281*3ba16761SJacob Faibussowitsch   }
282a297a907SKarl Rupp   if (specific) *specific = PetscErrorBaseMessage;
283*3ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
284e5c89e4eSSatish Balay }
285e5c89e4eSSatish Balay 
286984a1229SMatthew G. Knepley #if defined(PETSC_CLANGUAGE_CXX)
287984a1229SMatthew G. Knepley   /* C++ exceptions are formally not allowed to propagate through extern "C" code. In practice, far too much software
288984a1229SMatthew G. Knepley  * would be broken if implementations did not handle it it some common cases. However, keep in mind
289984a1229SMatthew G. Knepley  *
290984a1229SMatthew G. Knepley  *   Rule 62. Don't allow exceptions to propagate across module boundaries
291984a1229SMatthew G. Knepley  *
292984a1229SMatthew G. Knepley  * in "C++ Coding Standards" by Sutter and Alexandrescu. (This accounts for part of the ongoing C++ binary interface
293984a1229SMatthew G. Knepley  * instability.) Having PETSc raise errors as C++ exceptions was probably misguided and should eventually be removed.
294984a1229SMatthew G. Knepley  *
295984a1229SMatthew G. Knepley  * Here is the problem: You have a C++ function call a PETSc function, and you would like to maintain the error message
296984a1229SMatthew G. Knepley  * and stack information from the PETSc error. You could make everyone write exactly this code in their C++, but that
297984a1229SMatthew G. Knepley  * seems crazy to me.
298984a1229SMatthew G. Knepley  */
299984a1229SMatthew G. Knepley   #include <sstream>
3004c94c282SMatthew G. Knepley   #include <stdexcept>
301d71ae5a4SJacob Faibussowitsch static void PetscCxxErrorThrow()
302d71ae5a4SJacob Faibussowitsch {
303984a1229SMatthew G. Knepley   const char *str;
304984a1229SMatthew G. Knepley   if (eh && eh->ctx) {
305984a1229SMatthew G. Knepley     std::ostringstream *msg;
306984a1229SMatthew G. Knepley     msg = (std::ostringstream *)eh->ctx;
307984a1229SMatthew G. Knepley     str = msg->str().c_str();
308984a1229SMatthew G. Knepley   } else str = "Error detected in C PETSc";
309984a1229SMatthew G. Knepley 
310984a1229SMatthew G. Knepley   throw std::runtime_error(str);
311984a1229SMatthew G. Knepley }
312984a1229SMatthew G. Knepley #endif
313984a1229SMatthew G. Knepley 
314e5c89e4eSSatish Balay /*@C
31545b666d6SBarry Smith    PetscError - Routine that is called when an error has been detected, usually called through the macro SETERRQ(PETSC_COMM_SELF,).
316e5c89e4eSSatish Balay 
317811af0c4SBarry Smith   Collective
318e5c89e4eSSatish Balay 
319e5c89e4eSSatish Balay    Input Parameters:
320e32f2f54SBarry Smith +  comm - communicator over which error occurred.  ALL ranks of this communicator MUST call this routine
321e32f2f54SBarry Smith .  line - the line number of the error (indicated by __LINE__)
3223de71b31SHong Zhang .  func - the function name in which the error was detected
323e5c89e4eSSatish Balay .  file - the file in which the error was detected (indicated by __FILE__)
324e5c89e4eSSatish Balay .  n - the generic error number
325811af0c4SBarry Smith .  p - `PETSC_ERROR_INITIAL` indicates the error was initially detected, `PETSC_ERROR_REPEAT` indicates this is a traceback from a previously detected error
326e5c89e4eSSatish Balay -  mess - formatted message string - aka printf
327e5c89e4eSSatish Balay 
328811af0c4SBarry Smith   Options Database Keys:
32945b666d6SBarry Smith +  -error_output_stdout - output the error messages to stdout instead of the default stderr
33045b666d6SBarry Smith -  -error_output_none - do not output the error messages
33145b666d6SBarry Smith 
332e5c89e4eSSatish Balay   Level: intermediate
333e5c89e4eSSatish Balay 
334e5c89e4eSSatish Balay    Notes:
335e809461dSJacob Faibussowitsch    PETSc error handling is done with error return codes. A non-zero return indicates an error
336e809461dSJacob Faibussowitsch    was detected. The return-value of this routine is what is ultimately returned by
337e809461dSJacob Faibussowitsch    `SETERRQ()`.
33845b666d6SBarry Smith 
339e809461dSJacob Faibussowitsch    Note that numerical errors (potential divide by zero, for example) are not managed by the
340e809461dSJacob Faibussowitsch    error return codes; they are managed via, for example, `KSPGetConvergedReason()` that
341e809461dSJacob Faibussowitsch    indicates if the solve was successful or not. The option `-ksp_error_if_not_converged`, for
342e809461dSJacob Faibussowitsch    example, turns numerical failures into hard errors managed via `PetscError()`.
34345b666d6SBarry Smith 
344e809461dSJacob Faibussowitsch    PETSc provides a rich supply of error handlers, see the list below, and users can also
345e809461dSJacob Faibussowitsch    provide their own error handlers.
346811af0c4SBarry Smith 
347e809461dSJacob Faibussowitsch    If the user sets their own error handler (via `PetscPushErrorHandler()`) they may return any
348e809461dSJacob Faibussowitsch    arbitrary value from it, but are encouraged to return nonzero values. If the return value is
349e809461dSJacob Faibussowitsch    zero, `SETERRQ()` will ignore the value and return `PETSC_ERR_RETURN` (a nonzero value)
350e809461dSJacob Faibussowitsch    instead.
351e809461dSJacob Faibussowitsch 
352e809461dSJacob Faibussowitsch    Most users need not directly use this routine and the error handlers, but can instead use
353e809461dSJacob Faibussowitsch    the simplified interface `PetscCall()` or `SETERRQ()`.
354e5c89e4eSSatish Balay 
355e3081792SBarry Smith    Fortran Note:
356e3081792SBarry Smith    This routine is used differently from Fortran
357e3081792SBarry Smith $    PetscError(MPI_Comm comm,PetscErrorCode n,PetscErrorType p,char *message)
358e3081792SBarry Smith 
359811af0c4SBarry Smith    Developer Note:
360811af0c4SBarry Smith    Since this is called after an error condition it should not be calling any error handlers (currently it ignores any error codes)
3613bf036e2SBarry 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
3623bf036e2SBarry Smith    but this annoying.
3633bf036e2SBarry Smith 
364db781477SPatrick Sanan .seealso: `PetscErrorCode`, `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`,
365db781477SPatrick Sanan           `PetscReturnErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscEmacsClientErrorHandler()`,
366db781477SPatrick Sanan           `SETERRQ()`, `PetscCall()`, `CHKMEMQ`, `SETERRQ()`, `SETERRQ()`, `PetscErrorMessage()`, `PETSCABORT()`
367e5c89e4eSSatish Balay @*/
368d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscError(MPI_Comm comm, int line, const char *func, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, ...)
369d71ae5a4SJacob Faibussowitsch {
370e5c89e4eSSatish Balay   va_list        Argp;
371c9a19010SBarry Smith   size_t         fullLength;
37202c9f0b5SLisandro Dalcin   char           buf[2048], *lbuf = NULL;
373fbfcfee5SBarry Smith   PetscBool      ismain;
3743bf036e2SBarry Smith   PetscErrorCode ierr;
375e5c89e4eSSatish Balay 
37639a651e2SJacob Faibussowitsch   if (!PetscErrorHandlingInitialized) return n;
3770d577829SBarry Smith   if (comm == MPI_COMM_NULL) comm = PETSC_COMM_SELF;
378e5c89e4eSSatish Balay 
379e5c89e4eSSatish Balay   /* Compose the message evaluating the print format */
380e5c89e4eSSatish Balay   if (mess) {
381e5c89e4eSSatish Balay     va_start(Argp, mess);
382*3ba16761SJacob Faibussowitsch     ierr = PetscVSNPrintf(buf, 2048, mess, &fullLength, Argp);
383e5c89e4eSSatish Balay     va_end(Argp);
384e5c89e4eSSatish Balay     lbuf = buf;
385*3ba16761SJacob Faibussowitsch     if (p == PETSC_ERROR_INITIAL) ierr = PetscStrncpy(PetscErrorBaseMessage, lbuf, 1023);
386e5c89e4eSSatish Balay   }
387e5c89e4eSSatish Balay 
388*3ba16761SJacob Faibussowitsch   if (p == PETSC_ERROR_INITIAL && n != PETSC_ERR_MEMC) ierr = PetscMallocValidate(__LINE__, PETSC_FUNCTION_NAME, __FILE__);
3894ed0ab5bSBarry Smith 
39002c9f0b5SLisandro Dalcin   if (!eh) ierr = PetscTraceBackErrorHandler(comm, line, func, file, n, p, lbuf, NULL);
391efca3c55SSatish Balay   else ierr = (*eh->handler)(comm, line, func, file, n, p, lbuf, eh->ctx);
392362febeeSStefano Zampini   PetscStackClearTop;
393e5c89e4eSSatish Balay 
394e5c89e4eSSatish Balay   /*
3957233276eSBarry Smith       If this is called from the main() routine we call MPI_Abort() instead of
396e5c89e4eSSatish Balay     return to allow the parallel program to be properly shutdown.
397e5c89e4eSSatish Balay 
3987233276eSBarry Smith     Does not call PETSCABORT() since that would provide the wrong source file and line number information
399e5c89e4eSSatish Balay   */
40049c86fc7SBarry Smith   if (func) {
401*3ba16761SJacob Faibussowitsch     PetscErrorCode cmp_ierr = PetscStrncmp(func, "main", 4, &ismain);
4027233276eSBarry Smith     if (ismain) {
403*3ba16761SJacob Faibussowitsch       if (petscwaitonerrorflg) cmp_ierr = PetscSleep(1000);
404*3ba16761SJacob Faibussowitsch       (void)cmp_ierr;
405660278c0SBarry Smith       PETSCABORT(comm, ierr);
4067233276eSBarry Smith     }
40749c86fc7SBarry Smith   }
4082c280183SJed Brown #if defined(PETSC_CLANGUAGE_CXX)
409ad540459SPierre Jolivet   if (p == PETSC_ERROR_IN_CXX) PetscCxxErrorThrow();
410d736bfebSBarry Smith #endif
411362febeeSStefano Zampini   return ierr;
412e5c89e4eSSatish Balay }
413e5c89e4eSSatish Balay 
414e5c89e4eSSatish Balay /* -------------------------------------------------------------------------*/
415e5c89e4eSSatish Balay 
416e5c89e4eSSatish Balay /*@C
417e5c89e4eSSatish Balay     PetscIntView - Prints an array of integers; useful for debugging.
418e5c89e4eSSatish Balay 
419c3339decSBarry Smith     Collective
420e5c89e4eSSatish Balay 
421e5c89e4eSSatish Balay     Input Parameters:
422e5c89e4eSSatish Balay +   N - number of integers in array
423e5c89e4eSSatish Balay .   idx - array of integers
424811af0c4SBarry Smith -   viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
425e5c89e4eSSatish Balay 
426e5c89e4eSSatish Balay   Level: intermediate
427e5c89e4eSSatish Balay 
428811af0c4SBarry Smith     Note:
429811af0c4SBarry Smith     This may be called from within the debugger
430300a7f5bSBarry Smith 
431811af0c4SBarry Smith     Developer Note:
432811af0c4SBarry Smith     idx cannot be const because may be passed to binary viewer where temporary byte swapping may be done
433811af0c4SBarry Smith 
434811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscRealView()`
435e5c89e4eSSatish Balay @*/
436d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscIntView(PetscInt N, const PetscInt idx[], PetscViewer viewer)
437d71ae5a4SJacob Faibussowitsch {
438ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
439e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 20, p = N % 20;
440ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
441e5c89e4eSSatish Balay   MPI_Comm    comm;
442e5c89e4eSSatish Balay 
443e5c89e4eSSatish Balay   PetscFunctionBegin;
444e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
4450700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
4465ab33896SBarry Smith   if (N) PetscValidIntPointer(idx, 2);
4479566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
4489566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
4499566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
450e5c89e4eSSatish Balay 
4519566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
4529566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
453e5c89e4eSSatish Balay   if (iascii) {
4549566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
455e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
456ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4579566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * i));
458ca0c3be5SJacob Faibussowitsch       } else {
4599566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * i));
460ca0c3be5SJacob Faibussowitsch       }
46148a46eb9SPierre Jolivet       for (j = 0; j < 20; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[i * 20 + j]));
4629566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
463e5c89e4eSSatish Balay     }
464e5c89e4eSSatish Balay     if (p) {
465ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4669566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * n));
467ca0c3be5SJacob Faibussowitsch       } else {
4689566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * n));
469ca0c3be5SJacob Faibussowitsch       }
4709566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[20 * n + i]));
4719566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
472e5c89e4eSSatish Balay     }
4739566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
4749566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
4756805f65bSBarry Smith   } else if (isbinary) {
476ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
477e5c89e4eSSatish Balay     PetscInt    *array;
478783b601eSJed Brown 
4799566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
480e5c89e4eSSatish Balay 
481e5c89e4eSSatish Balay     if (size > 1) {
482e5c89e4eSSatish Balay       if (rank) {
4839566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
4849566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, NULL, NULL, NULL, MPIU_INT, 0, comm));
485e5c89e4eSSatish Balay       } else {
4869566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
4879566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
488e5c89e4eSSatish Balay         Ntotal = sizes[0];
4899566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
490e5c89e4eSSatish Balay         displs[0] = 0;
491e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
492e5c89e4eSSatish Balay           Ntotal += sizes[i];
493e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
494e5c89e4eSSatish Balay         }
4959566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
4969566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, array, sizes, displs, MPIU_INT, 0, comm));
4979566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_INT));
4989566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
4999566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
5009566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
501e5c89e4eSSatish Balay       }
502e5c89e4eSSatish Balay     } else {
5039566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, idx, N, PETSC_INT));
504e5c89e4eSSatish Balay     }
505e5c89e4eSSatish Balay   } else {
506e5c89e4eSSatish Balay     const char *tname;
5079566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
50898921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
509e5c89e4eSSatish Balay   }
510*3ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
511e5c89e4eSSatish Balay }
512e5c89e4eSSatish Balay 
513e5c89e4eSSatish Balay /*@C
514e5c89e4eSSatish Balay     PetscRealView - Prints an array of doubles; useful for debugging.
515e5c89e4eSSatish Balay 
516c3339decSBarry Smith     Collective
517e5c89e4eSSatish Balay 
518e5c89e4eSSatish Balay     Input Parameters:
519811af0c4SBarry Smith +   N - number of `PetscReal` in array
520811af0c4SBarry Smith .   idx - array of `PetscReal`
521811af0c4SBarry Smith -   viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
522e5c89e4eSSatish Balay 
523e5c89e4eSSatish Balay   Level: intermediate
524e5c89e4eSSatish Balay 
525811af0c4SBarry Smith     Note:
526811af0c4SBarry Smith     This may be called from within the debugger
527300a7f5bSBarry Smith 
528811af0c4SBarry Smith     Developer Note:
529811af0c4SBarry Smith     idx cannot be const because may be passed to binary viewer where temporary byte swapping may be done
530811af0c4SBarry Smith 
531811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`
532e5c89e4eSSatish Balay @*/
533d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscRealView(PetscInt N, const PetscReal idx[], PetscViewer viewer)
534d71ae5a4SJacob Faibussowitsch {
535ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
536e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 5, p = N % 5;
537ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
538e5c89e4eSSatish Balay   MPI_Comm    comm;
539e5c89e4eSSatish Balay 
540e5c89e4eSSatish Balay   PetscFunctionBegin;
541e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
5420700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
543064a246eSJacob Faibussowitsch   PetscValidRealPointer(idx, 2);
5449566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
5459566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
5469566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
547e5c89e4eSSatish Balay 
5489566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
5499566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
550e5c89e4eSSatish Balay   if (iascii) {
5511a989b97SToby Isaac     PetscInt tab;
5521a989b97SToby Isaac 
5539566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
5549566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIGetTab(viewer, &tab));
555e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
5569566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
557ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5589566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * i));
559ca0c3be5SJacob Faibussowitsch       } else {
5609566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * i));
561ca0c3be5SJacob Faibussowitsch       }
5629566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
56348a46eb9SPierre Jolivet       for (j = 0; j < 5; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 5 + j]));
5649566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
565e5c89e4eSSatish Balay     }
566e5c89e4eSSatish Balay     if (p) {
5679566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
568ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5699566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * n));
570ca0c3be5SJacob Faibussowitsch       } else {
5719566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * n));
572ca0c3be5SJacob Faibussowitsch       }
5739566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
5749566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[5 * n + i]));
5759566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
576e5c89e4eSSatish Balay     }
5779566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
5789566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIISetTab(viewer, tab));
5799566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
5806805f65bSBarry Smith   } else if (isbinary) {
581ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, *displs, Ntotal, NN;
582e5c89e4eSSatish Balay     PetscReal   *array;
583e5c89e4eSSatish Balay 
5849566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
585e5c89e4eSSatish Balay 
586e5c89e4eSSatish Balay     if (size > 1) {
587e5c89e4eSSatish Balay       if (rank) {
5889566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
5899566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, NULL, NULL, NULL, MPIU_REAL, 0, comm));
590e5c89e4eSSatish Balay       } else {
5919566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
5929566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
593e5c89e4eSSatish Balay         Ntotal = sizes[0];
5949566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
595e5c89e4eSSatish Balay         displs[0] = 0;
596e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
597e5c89e4eSSatish Balay           Ntotal += sizes[i];
598e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
599e5c89e4eSSatish Balay         }
6009566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
6019566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, array, sizes, displs, MPIU_REAL, 0, comm));
6029566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_REAL));
6039566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
6049566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
6059566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
606e5c89e4eSSatish Balay       }
607e5c89e4eSSatish Balay     } else {
6089566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_REAL));
609e5c89e4eSSatish Balay     }
610e5c89e4eSSatish Balay   } else {
611e5c89e4eSSatish Balay     const char *tname;
6129566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
61398921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
614e5c89e4eSSatish Balay   }
615*3ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
616e5c89e4eSSatish Balay }
617e5c89e4eSSatish Balay 
618e5c89e4eSSatish Balay /*@C
619811af0c4SBarry Smith     PetscScalarView - Prints an array of `PetscScalar`; useful for debugging.
620e5c89e4eSSatish Balay 
621c3339decSBarry Smith     Collective
622e5c89e4eSSatish Balay 
623e5c89e4eSSatish Balay     Input Parameters:
624e5c89e4eSSatish Balay +   N - number of scalars in array
625e5c89e4eSSatish Balay .   idx - array of scalars
626811af0c4SBarry Smith -   viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
627e5c89e4eSSatish Balay 
628e5c89e4eSSatish Balay   Level: intermediate
629e5c89e4eSSatish Balay 
630811af0c4SBarry Smith     Note:
631811af0c4SBarry Smith     This may be called from within the debugger
632300a7f5bSBarry Smith 
633811af0c4SBarry Smith     Developer Note:
634811af0c4SBarry Smith     idx cannot be const because may be passed to binary viewer where byte swapping may be done
635811af0c4SBarry Smith 
636811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`, `PetscRealView()`
637e5c89e4eSSatish Balay @*/
638d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscScalarView(PetscInt N, const PetscScalar idx[], PetscViewer viewer)
639d71ae5a4SJacob Faibussowitsch {
640ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
641e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 3, p = N % 3;
642ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
643e5c89e4eSSatish Balay   MPI_Comm    comm;
644e5c89e4eSSatish Balay 
645e5c89e4eSSatish Balay   PetscFunctionBegin;
646e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
647e5c89e4eSSatish Balay   PetscValidHeader(viewer, 3);
6488c34849bSStefano Zampini   if (N) PetscValidScalarPointer(idx, 2);
6499566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
6509566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
6519566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
652e5c89e4eSSatish Balay 
6539566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
6549566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
655e5c89e4eSSatish Balay   if (iascii) {
6569566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
657e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
658ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6599566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * i));
660ca0c3be5SJacob Faibussowitsch       } else {
6619566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * i));
662ca0c3be5SJacob Faibussowitsch       }
663e5c89e4eSSatish Balay       for (j = 0; j < 3; j++) {
664e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6659566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[i * 3 + j]), (double)PetscImaginaryPart(idx[i * 3 + j])));
666e5c89e4eSSatish Balay #else
6679566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 3 + j]));
668e5c89e4eSSatish Balay #endif
669e5c89e4eSSatish Balay       }
6709566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
671e5c89e4eSSatish Balay     }
672e5c89e4eSSatish Balay     if (p) {
673ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6749566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * n));
675ca0c3be5SJacob Faibussowitsch       } else {
6769566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * n));
677ca0c3be5SJacob Faibussowitsch       }
678e5c89e4eSSatish Balay       for (i = 0; i < p; i++) {
679e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6809566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[n * 3 + i]), (double)PetscImaginaryPart(idx[n * 3 + i])));
681e5c89e4eSSatish Balay #else
6829566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[3 * n + i]));
683e5c89e4eSSatish Balay #endif
684e5c89e4eSSatish Balay       }
6859566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
686e5c89e4eSSatish Balay     }
6879566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
6889566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
6896805f65bSBarry Smith   } else if (isbinary) {
690ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
691e5c89e4eSSatish Balay     PetscScalar *array;
692e5c89e4eSSatish Balay 
6939566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
694e5c89e4eSSatish Balay 
695e5c89e4eSSatish Balay     if (size > 1) {
696e5c89e4eSSatish Balay       if (rank) {
6979566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
6989566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, NULL, NULL, NULL, MPIU_SCALAR, 0, comm));
699e5c89e4eSSatish Balay       } else {
7009566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
7019566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
702e5c89e4eSSatish Balay         Ntotal = sizes[0];
7039566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
704e5c89e4eSSatish Balay         displs[0] = 0;
705e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
706e5c89e4eSSatish Balay           Ntotal += sizes[i];
707e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
708e5c89e4eSSatish Balay         }
7099566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
7109566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, array, sizes, displs, MPIU_SCALAR, 0, comm));
7119566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_SCALAR));
7129566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
7139566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
7149566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
715e5c89e4eSSatish Balay       }
716e5c89e4eSSatish Balay     } else {
7179566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_SCALAR));
718e5c89e4eSSatish Balay     }
719e5c89e4eSSatish Balay   } else {
720e5c89e4eSSatish Balay     const char *tname;
7219566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
72298921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
723e5c89e4eSSatish Balay   }
724*3ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
725e5c89e4eSSatish Balay }
726e5c89e4eSSatish Balay 
727e22e20c5SJunchao Zhang #if defined(PETSC_HAVE_CUDA)
7280e6b6b59SJacob Faibussowitsch   #include <petscdevice_cuda.h>
729d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUBLASGetErrorName(cublasStatus_t status)
730d71ae5a4SJacob Faibussowitsch {
731e22e20c5SJunchao Zhang   switch (status) {
732e22e20c5SJunchao Zhang   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
733d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_SUCCESS:
734d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_SUCCESS";
735d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_INITIALIZED:
736d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_INITIALIZED";
737d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ALLOC_FAILED:
738d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ALLOC_FAILED";
739d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INVALID_VALUE:
740d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INVALID_VALUE";
741d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ARCH_MISMATCH:
742d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ARCH_MISMATCH";
743d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_MAPPING_ERROR:
744d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_MAPPING_ERROR";
745d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_EXECUTION_FAILED:
746d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_EXECUTION_FAILED";
747d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INTERNAL_ERROR:
748d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INTERNAL_ERROR";
749d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_SUPPORTED:
750d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_SUPPORTED";
751d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_LICENSE_ERROR:
752d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_LICENSE_ERROR";
753e22e20c5SJunchao Zhang   #endif
754d71ae5a4SJacob Faibussowitsch   default:
755d71ae5a4SJacob Faibussowitsch     return "unknown error";
756e22e20c5SJunchao Zhang   }
757e22e20c5SJunchao Zhang }
758d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUSolverGetErrorName(cusolverStatus_t status)
759d71ae5a4SJacob Faibussowitsch {
760a4b895e1SBarry Smith   switch (status) {
761a4b895e1SBarry Smith   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
762d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_SUCCESS:
763d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_SUCCESS";
764d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_INITIALIZED:
765d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_INITIALIZED";
766d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_VALUE:
767d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_VALUE";
768d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ARCH_MISMATCH:
769d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ARCH_MISMATCH";
770d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INTERNAL_ERROR:
771d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INTERNAL_ERROR";
772030f984aSJacob Faibussowitsch     #if (CUDART_VERSION >= 9000) /* CUDA 9.0 had these defined on June 2021 */
773d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ALLOC_FAILED:
774d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ALLOC_FAILED";
775d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MAPPING_ERROR:
776d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MAPPING_ERROR";
777d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_EXECUTION_FAILED:
778d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_EXECUTION_FAILED";
779d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
780d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
781d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_SUPPORTED:
782d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_SUPPORTED ";
783d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ZERO_PIVOT:
784d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ZERO_PIVOT";
785d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_LICENSE:
786d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_LICENSE";
787a4b895e1SBarry Smith     #endif
788030f984aSJacob Faibussowitsch   #endif
789d71ae5a4SJacob Faibussowitsch   default:
790d71ae5a4SJacob Faibussowitsch     return "unknown error";
791030f984aSJacob Faibussowitsch   }
792030f984aSJacob Faibussowitsch }
793d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUFFTGetErrorName(cufftResult result)
794d71ae5a4SJacob Faibussowitsch {
795030f984aSJacob Faibussowitsch   switch (result) {
796d71ae5a4SJacob Faibussowitsch   case CUFFT_SUCCESS:
797d71ae5a4SJacob Faibussowitsch     return "CUFFT_SUCCESS";
798d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_PLAN:
799d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_PLAN";
800d71ae5a4SJacob Faibussowitsch   case CUFFT_ALLOC_FAILED:
801d71ae5a4SJacob Faibussowitsch     return "CUFFT_ALLOC_FAILED";
802d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_TYPE:
803d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_TYPE";
804d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_VALUE:
805d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_VALUE";
806d71ae5a4SJacob Faibussowitsch   case CUFFT_INTERNAL_ERROR:
807d71ae5a4SJacob Faibussowitsch     return "CUFFT_INTERNAL_ERROR";
808d71ae5a4SJacob Faibussowitsch   case CUFFT_EXEC_FAILED:
809d71ae5a4SJacob Faibussowitsch     return "CUFFT_EXEC_FAILED";
810d71ae5a4SJacob Faibussowitsch   case CUFFT_SETUP_FAILED:
811d71ae5a4SJacob Faibussowitsch     return "CUFFT_SETUP_FAILED";
812d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_SIZE:
813d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_SIZE";
814d71ae5a4SJacob Faibussowitsch   case CUFFT_UNALIGNED_DATA:
815d71ae5a4SJacob Faibussowitsch     return "CUFFT_UNALIGNED_DATA";
816d71ae5a4SJacob Faibussowitsch   case CUFFT_INCOMPLETE_PARAMETER_LIST:
817d71ae5a4SJacob Faibussowitsch     return "CUFFT_INCOMPLETE_PARAMETER_LIST";
818d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_DEVICE:
819d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_DEVICE";
820d71ae5a4SJacob Faibussowitsch   case CUFFT_PARSE_ERROR:
821d71ae5a4SJacob Faibussowitsch     return "CUFFT_PARSE_ERROR";
822d71ae5a4SJacob Faibussowitsch   case CUFFT_NO_WORKSPACE:
823d71ae5a4SJacob Faibussowitsch     return "CUFFT_NO_WORKSPACE";
824d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_IMPLEMENTED:
825d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_IMPLEMENTED";
826d71ae5a4SJacob Faibussowitsch   case CUFFT_LICENSE_ERROR:
827d71ae5a4SJacob Faibussowitsch     return "CUFFT_LICENSE_ERROR";
828d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_SUPPORTED:
829d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_SUPPORTED";
830d71ae5a4SJacob Faibussowitsch   default:
831d71ae5a4SJacob Faibussowitsch     return "unknown error";
832a4b895e1SBarry Smith   }
833a4b895e1SBarry Smith }
834e22e20c5SJunchao Zhang #endif
83559af0bd3SScott Kruger 
83659af0bd3SScott Kruger #if defined(PETSC_HAVE_HIP)
8370e6b6b59SJacob Faibussowitsch   #include <petscdevice_hip.h>
838d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscHIPBLASGetErrorName(hipblasStatus_t status)
839d71ae5a4SJacob Faibussowitsch {
84059af0bd3SScott Kruger   switch (status) {
841d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_SUCCESS:
842d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_SUCCESS";
843d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_INITIALIZED:
844d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_INITIALIZED";
845d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ALLOC_FAILED:
846d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ALLOC_FAILED";
847d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INVALID_VALUE:
848d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INVALID_VALUE";
849d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ARCH_MISMATCH:
850d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ARCH_MISMATCH";
851d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_MAPPING_ERROR:
852d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_MAPPING_ERROR";
853d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_EXECUTION_FAILED:
854d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_EXECUTION_FAILED";
855d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INTERNAL_ERROR:
856d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INTERNAL_ERROR";
857d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_SUPPORTED:
858d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_SUPPORTED";
859d71ae5a4SJacob Faibussowitsch   default:
860d71ae5a4SJacob Faibussowitsch     return "unknown error";
86159af0bd3SScott Kruger   }
86259af0bd3SScott Kruger }
86347d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSPARSEGetErrorName(hipsparseStatus_t status)
86447d993e7Ssuyashtn {
86547d993e7Ssuyashtn   switch (status) {
86647d993e7Ssuyashtn   case HIPSPARSE_STATUS_SUCCESS:
86747d993e7Ssuyashtn     return "HIPSPARSE_STATUS_SUCCESS";
86847d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_INITIALIZED:
86947d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_INITIALIZED";
87047d993e7Ssuyashtn   case HIPSPARSE_STATUS_ALLOC_FAILED:
87147d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ALLOC_FAILED";
87247d993e7Ssuyashtn   case HIPSPARSE_STATUS_INVALID_VALUE:
87347d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INVALID_VALUE";
87447d993e7Ssuyashtn   case HIPSPARSE_STATUS_ARCH_MISMATCH:
87547d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ARCH_MISMATCH";
87647d993e7Ssuyashtn   case HIPSPARSE_STATUS_MAPPING_ERROR:
87747d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MAPPING_ERROR";
87847d993e7Ssuyashtn   case HIPSPARSE_STATUS_EXECUTION_FAILED:
87947d993e7Ssuyashtn     return "HIPSPARSE_STATUS_EXECUTION_FAILED";
88047d993e7Ssuyashtn   case HIPSPARSE_STATUS_INTERNAL_ERROR:
88147d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INTERNAL_ERROR";
88247d993e7Ssuyashtn   case HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
88347d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
88447d993e7Ssuyashtn   case HIPSPARSE_STATUS_ZERO_PIVOT:
88547d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ZERO_PIVOT";
88647d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_SUPPORTED:
88747d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_SUPPORTED";
88847d993e7Ssuyashtn   case HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES:
88947d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES";
89047d993e7Ssuyashtn   default:
89147d993e7Ssuyashtn     return "unknown error";
89247d993e7Ssuyashtn   }
89347d993e7Ssuyashtn }
89447d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSolverGetErrorName(hipsolverStatus_t status)
89547d993e7Ssuyashtn {
89647d993e7Ssuyashtn   switch (status) {
89747d993e7Ssuyashtn   case HIPSOLVER_STATUS_SUCCESS:
89847d993e7Ssuyashtn     return "HIPSOLVER_STATUS_SUCCESS";
89947d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_INITIALIZED:
90047d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_INITIALIZED";
90147d993e7Ssuyashtn   case HIPSOLVER_STATUS_ALLOC_FAILED:
90247d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ALLOC_FAILED";
90347d993e7Ssuyashtn   case HIPSOLVER_STATUS_MAPPING_ERROR:
90447d993e7Ssuyashtn     return "HIPSOLVER_STATUS_MAPPING_ERROR";
90547d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_VALUE:
90647d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_VALUE";
90747d993e7Ssuyashtn   case HIPSOLVER_STATUS_EXECUTION_FAILED:
90847d993e7Ssuyashtn     return "HIPSOLVER_STATUS_EXECUTION_FAILED";
90947d993e7Ssuyashtn   case HIPSOLVER_STATUS_INTERNAL_ERROR:
91047d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INTERNAL_ERROR";
91147d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_SUPPORTED:
91247d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_SUPPORTED ";
91347d993e7Ssuyashtn   case HIPSOLVER_STATUS_ARCH_MISMATCH:
91447d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ARCH_MISMATCH";
91547d993e7Ssuyashtn   case HIPSOLVER_STATUS_HANDLE_IS_NULLPTR:
91647d993e7Ssuyashtn     return "HIPSOLVER_STATUS_HANDLE_IS_NULLPTR";
91747d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_ENUM:
91847d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_ENUM";
91947d993e7Ssuyashtn   case HIPSOLVER_STATUS_UNKNOWN:
92047d993e7Ssuyashtn   default:
92147d993e7Ssuyashtn     return "HIPSOLVER_STATUS_UNKNOWN";
92247d993e7Ssuyashtn   }
92347d993e7Ssuyashtn }
92459af0bd3SScott Kruger #endif
925db9cea48SBarry Smith 
926db9cea48SBarry Smith /*@
927811af0c4SBarry Smith       PetscMPIErrorString - Given an MPI error code returns the `MPI_Error_string()` appropriately
928db9cea48SBarry Smith            formatted for displaying with the PETSc error handlers.
929db9cea48SBarry Smith 
930db9cea48SBarry Smith  Input Parameter:
931db9cea48SBarry Smith .  err - the MPI error code
932db9cea48SBarry Smith 
933db9cea48SBarry Smith  Output Parameter:
934811af0c4SBarry Smith .  string - the MPI error message, should declare its length to be larger than `MPI_MAX_ERROR_STRING`
935db9cea48SBarry Smith 
93649c86fc7SBarry Smith    Level: developer
93749c86fc7SBarry Smith 
938811af0c4SBarry Smith  Note:
939db9cea48SBarry Smith     Does not return an error code or do error handling because it may be called from inside an error handler
940db9cea48SBarry Smith 
941db9cea48SBarry Smith @*/
942d71ae5a4SJacob Faibussowitsch void PetscMPIErrorString(PetscMPIInt err, char *string)
943d71ae5a4SJacob Faibussowitsch {
944db9cea48SBarry Smith   char        errorstring[MPI_MAX_ERROR_STRING];
945db9cea48SBarry Smith   PetscMPIInt len, j = 0;
946db9cea48SBarry Smith 
947db9cea48SBarry Smith   MPI_Error_string(err, (char *)errorstring, &len);
948db9cea48SBarry Smith   for (PetscMPIInt i = 0; i < len; i++) {
949db9cea48SBarry Smith     string[j++] = errorstring[i];
950db9cea48SBarry Smith     if (errorstring[i] == '\n') {
951db9cea48SBarry Smith       for (PetscMPIInt k = 0; k < 16; k++) string[j++] = ' ';
952db9cea48SBarry Smith     }
953db9cea48SBarry Smith   }
954db9cea48SBarry Smith   string[j] = 0;
955db9cea48SBarry Smith }
956