xref: /petsc/src/sys/error/err.c (revision dd460d279d400be05a060420ac00c7301eb4da3f)
1e5c89e4eSSatish Balay /*
2e5c89e4eSSatish Balay       Code that allows one to set the error handlers
347d993e7Ssuyashtn       Portions of this code are under:
447d993e7Ssuyashtn       Copyright (c) 2022 Advanced Micro Devices, Inc. All rights reserved.
5e5c89e4eSSatish Balay */
6af0996ceSBarry Smith #include <petsc/private/petscimpl.h> /*I "petscsys.h" I*/
7665c2dedSJed Brown #include <petscviewer.h>
8e5c89e4eSSatish Balay 
9e5c89e4eSSatish Balay typedef struct _EH *EH;
10e5c89e4eSSatish Balay struct _EH {
11efca3c55SSatish Balay   PetscErrorCode (*handler)(MPI_Comm, int, const char *, const char *, PetscErrorCode, PetscErrorType, const char *, void *);
12e5c89e4eSSatish Balay   void *ctx;
13e5c89e4eSSatish Balay   EH    previous;
14e5c89e4eSSatish Balay };
15e5c89e4eSSatish Balay 
1635f00c14SToby Isaac /* This is here to allow the traceback error handler (or potentially other error handlers)
1735f00c14SToby Isaac    to certify that PETSCABORT is being called on all MPI processes, and that it should be possible to call
1835f00c14SToby Isaac    MPI_Finalize() and exit().  This should only be used when `PetscCIEnabledPortabeErrorOutput == PETSC_TRUE`
1935f00c14SToby Isaac    to allow testing of error messages.  Do not rely on this for clean exit in production. */
2035f00c14SToby Isaac PetscBool petscabortmpifinalize = PETSC_FALSE;
2135f00c14SToby Isaac 
2202c9f0b5SLisandro Dalcin static EH eh = NULL;
23e5c89e4eSSatish Balay 
24e5c89e4eSSatish Balay /*@C
25e5c89e4eSSatish Balay   PetscEmacsClientErrorHandler - Error handler that uses the emacsclient program to
26a5b23f4aSJose E. Roman   load the file where the error occurred. Then calls the "previous" error handler.
27e5c89e4eSSatish Balay 
28cc4c1da9SBarry Smith   Not Collective, No Fortran Support
29e5c89e4eSSatish Balay 
30e5c89e4eSSatish Balay   Input Parameters:
31a5b23f4aSJose E. Roman + comm - communicator over which error occurred
326e25c4a1SBarry Smith . line - the line number of the error (usually indicated by `__LINE__` in the calling routine)
336e25c4a1SBarry Smith . file - the file in which the error was detected (usually indicated by `__FILE__` in the calling routine)
346e25c4a1SBarry Smith . fun  - the function name of the calling routine
35e5c89e4eSSatish Balay . mess - an error text string, usually just printed to the screen
36e5c89e4eSSatish Balay . n    - the generic error number
376e25c4a1SBarry Smith . p    - `PETSC_ERROR_INITIAL` indicates this is the first time the error handler is being called while `PETSC_ERROR_REPEAT` indicates it was previously called
38e5c89e4eSSatish Balay - ctx  - error handler context
39e5c89e4eSSatish Balay 
40e5c89e4eSSatish Balay   Options Database Key:
4110699b91SBarry Smith . -on_error_emacs <machinename> - will contact machinename to open the Emacs client there
42e5c89e4eSSatish Balay 
43e5c89e4eSSatish Balay   Level: developer
44e5c89e4eSSatish Balay 
45811af0c4SBarry Smith   Note:
46e5c89e4eSSatish Balay   You must put (server-start) in your .emacs file for the emacsclient software to work
47e5c89e4eSSatish Balay 
486e25c4a1SBarry Smith   Developer Note:
49811af0c4SBarry Smith   Since this is an error handler it cannot call `PetscCall()`; thus we just return if an error is detected.
50811af0c4SBarry Smith   But some of the functions it calls do perform error checking that may not be appropriate in a error handler call.
513bf036e2SBarry Smith 
52db781477SPatrick Sanan .seealso: `PetscError()`, `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscAttachDebuggerErrorHandler()`,
536e25c4a1SBarry Smith           `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscReturnErrorHandler()`,
546e25c4a1SBarry Smith           `PetscErrorType`, `PETSC_ERROR_INITIAL`, `PETSC_ERROR_REPEAT`, `PetscErrorCode`
55e5c89e4eSSatish Balay  @*/
56d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscEmacsClientErrorHandler(MPI_Comm comm, int line, const char *fun, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, void *ctx)
57d71ae5a4SJacob Faibussowitsch {
58e5c89e4eSSatish Balay   PetscErrorCode ierr;
59e5c89e4eSSatish Balay   char           command[PETSC_MAX_PATH_LEN];
60e5c89e4eSSatish Balay   const char    *pdir;
61e5c89e4eSSatish Balay   FILE          *fp;
62e5c89e4eSSatish Balay 
639371c9d4SSatish Balay   ierr = PetscGetPetscDir(&pdir);
6411cc89d2SBarry Smith   if (ierr) return ierr;
65a364092eSJacob Faibussowitsch   ierr = PetscSNPrintf(command, PETSC_STATIC_ARRAY_LENGTH(command), "cd %s; emacsclient --no-wait +%d %s\n", pdir, line, file);
66a364092eSJacob Faibussowitsch   if (ierr) return ierr;
67e5c89e4eSSatish Balay #if defined(PETSC_HAVE_POPEN)
689371c9d4SSatish Balay   ierr = PetscPOpen(MPI_COMM_WORLD, (char *)ctx, command, "r", &fp);
6911cc89d2SBarry Smith   if (ierr) return ierr;
709371c9d4SSatish Balay   ierr = PetscPClose(MPI_COMM_WORLD, fp);
7111cc89d2SBarry Smith   if (ierr) return ierr;
72e5c89e4eSSatish Balay #else
73e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP_SYS, "Cannot run external programs on this machine");
74e5c89e4eSSatish Balay #endif
759371c9d4SSatish Balay   ierr = PetscPopErrorHandler();
7611cc89d2SBarry Smith   if (ierr) return ierr; /* remove this handler from the stack of handlers */
773bf036e2SBarry Smith   if (!eh) {
789371c9d4SSatish Balay     ierr = PetscTraceBackErrorHandler(comm, line, fun, file, n, p, mess, NULL);
7911cc89d2SBarry Smith     if (ierr) return ierr;
803bf036e2SBarry Smith   } else {
819371c9d4SSatish Balay     ierr = (*eh->handler)(comm, line, fun, file, n, p, mess, eh->ctx);
8211cc89d2SBarry Smith     if (ierr) return ierr;
833bf036e2SBarry Smith   }
843ba16761SJacob Faibussowitsch   return PETSC_SUCCESS;
85e5c89e4eSSatish Balay }
86e5c89e4eSSatish Balay 
87e5c89e4eSSatish Balay /*@C
88e5c89e4eSSatish Balay   PetscPushErrorHandler - Sets a routine to be called on detection of errors.
89e5c89e4eSSatish Balay 
90cc4c1da9SBarry Smith   Not Collective, No Fortran Support
91e5c89e4eSSatish Balay 
92e5c89e4eSSatish Balay   Input Parameters:
93e5c89e4eSSatish Balay + handler - error handler routine
94e5c89e4eSSatish Balay - ctx     - optional handler context that contains information needed by the handler (for
95e5c89e4eSSatish Balay             example file pointers for error messages etc.)
96e5c89e4eSSatish Balay 
9720f4b53cSBarry Smith   Calling sequence of `handler`:
98a5b23f4aSJose E. Roman + comm - communicator over which error occurred
996e25c4a1SBarry Smith . line - the line number of the error (usually indicated by `__LINE__` in the calling routine)
1006e25c4a1SBarry Smith . file - the file in which the error was detected (usually indicated by `__FILE__` in the calling routine)
1016e25c4a1SBarry Smith . fun  - the function name of the calling routine
102e5c89e4eSSatish Balay . n    - the generic error number (see list defined in include/petscerror.h)
10320f4b53cSBarry Smith . p    - `PETSC_ERROR_INITIAL` if error just detected, otherwise `PETSC_ERROR_REPEAT`
104e5c89e4eSSatish Balay . mess - an error text string, usually just printed to the screen
105e5c89e4eSSatish Balay - ctx  - the error handler context
106e5c89e4eSSatish Balay 
107e5c89e4eSSatish Balay   Options Database Keys:
1086e25c4a1SBarry Smith + -on_error_attach_debugger <noxterm,lldb or gdb> - starts up the debugger if an error occurs
10910699b91SBarry Smith - -on_error_abort                                 - aborts the program if an error occurs
110e5c89e4eSSatish Balay 
111e5c89e4eSSatish Balay   Level: intermediate
112e5c89e4eSSatish Balay 
113811af0c4SBarry Smith   Note:
114811af0c4SBarry Smith   The currently available PETSc error handlers include `PetscTraceBackErrorHandler()`,
1156e25c4a1SBarry Smith   `PetscAttachDebuggerErrorHandler()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`, and `PetscReturnErrorHandler()`.
116e93bc3c1Svictor 
1176e25c4a1SBarry Smith   Fortran Note:
1186e25c4a1SBarry Smith   You can only push a single error handler from Fortran before popping it.
1197850c7c0SBarry Smith 
1206e25c4a1SBarry Smith .seealso: `PetscPopErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscAbortErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscPushSignalHandler()`,
1216e25c4a1SBarry Smith           `PetscErrorType`, `PETSC_ERROR_INITIAL`, `PETSC_ERROR_REPEAT`, `PetscErrorCode`
122e5c89e4eSSatish Balay @*/
12310450e9eSJacob Faibussowitsch PetscErrorCode PetscPushErrorHandler(PetscErrorCode (*handler)(MPI_Comm comm, int line, const char *fun, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, void *ctx), void *ctx)
124d71ae5a4SJacob Faibussowitsch {
125e5c89e4eSSatish Balay   EH neweh;
126e5c89e4eSSatish Balay 
127e5c89e4eSSatish Balay   PetscFunctionBegin;
1289566063dSJacob Faibussowitsch   PetscCall(PetscNew(&neweh));
129a297a907SKarl Rupp   if (eh) neweh->previous = eh;
13002c9f0b5SLisandro Dalcin   else neweh->previous = NULL;
131e5c89e4eSSatish Balay   neweh->handler = handler;
132e5c89e4eSSatish Balay   neweh->ctx     = ctx;
133e5c89e4eSSatish Balay   eh             = neweh;
1343ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
135e5c89e4eSSatish Balay }
136e5c89e4eSSatish Balay 
137e30d2299SSatish Balay /*@
138e5c89e4eSSatish Balay   PetscPopErrorHandler - Removes the latest error handler that was
139811af0c4SBarry Smith   pushed with `PetscPushErrorHandler()`.
140e5c89e4eSSatish Balay 
141e5c89e4eSSatish Balay   Not Collective
142e5c89e4eSSatish Balay 
143e5c89e4eSSatish Balay   Level: intermediate
144e5c89e4eSSatish Balay 
145db781477SPatrick Sanan .seealso: `PetscPushErrorHandler()`
146e5c89e4eSSatish Balay @*/
147d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscPopErrorHandler(void)
148d71ae5a4SJacob Faibussowitsch {
149e5c89e4eSSatish Balay   EH tmp;
150e5c89e4eSSatish Balay 
151e5c89e4eSSatish Balay   PetscFunctionBegin;
1523ba16761SJacob Faibussowitsch   if (!eh) PetscFunctionReturn(PETSC_SUCCESS);
153e5c89e4eSSatish Balay   tmp = eh;
154e5c89e4eSSatish Balay   eh  = eh->previous;
1559566063dSJacob Faibussowitsch   PetscCall(PetscFree(tmp));
1563ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
157e5c89e4eSSatish Balay }
158e5c89e4eSSatish Balay 
159e93bc3c1Svictor /*@C
16045b666d6SBarry Smith   PetscReturnErrorHandler - Error handler that causes a return without printing an error message.
161e93bc3c1Svictor 
162cc4c1da9SBarry Smith   Not Collective, No Fortran Support
163e93bc3c1Svictor 
164e93bc3c1Svictor   Input Parameters:
165e32f2f54SBarry Smith + comm - communicator over which error occurred
1666e25c4a1SBarry Smith . line - the line number of the error (usually indicated by `__LINE__` in the calling routine)
16710450e9eSJacob Faibussowitsch . fun  - the function name
1686e25c4a1SBarry Smith . file - the file in which the error was detected (usually indicated by `__FILE__` in the calling routine)
169e93bc3c1Svictor . mess - an error text string, usually just printed to the screen
170e93bc3c1Svictor . n    - the generic error number
1716e25c4a1SBarry Smith . p    - `PETSC_ERROR_INITIAL` indicates this is the first time the error handler is being called while `PETSC_ERROR_REPEAT` indicates it was previously called
172e93bc3c1Svictor - ctx  - error handler context
173e93bc3c1Svictor 
174e93bc3c1Svictor   Level: developer
175e93bc3c1Svictor 
176e93bc3c1Svictor   Notes:
1776e25c4a1SBarry Smith   Users do not directly employ this routine
178e93bc3c1Svictor 
1796e25c4a1SBarry Smith   Use `PetscPushErrorHandler()` to set the desired error handler.  The
1806e25c4a1SBarry Smith   currently available PETSc error handlers include `PetscTraceBackErrorHandler()`,
1816e25c4a1SBarry Smith   `PetscAttachDebuggerErrorHandler()`, and `PetscAbortErrorHandler()`.
182e93bc3c1Svictor 
183db781477SPatrick Sanan .seealso: `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscError()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`, `PetscTraceBackErrorHandler()`,
1846e25c4a1SBarry Smith           `PetscAttachDebuggerErrorHandler()`, `PetscEmacsClientErrorHandler()`,
1856e25c4a1SBarry Smith           `PetscErrorType`, `PETSC_ERROR_INITIAL`, `PETSC_ERROR_REPEAT`, `PetscErrorCode`
186e93bc3c1Svictor  @*/
187d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscReturnErrorHandler(MPI_Comm comm, int line, const char *fun, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, void *ctx)
188d71ae5a4SJacob Faibussowitsch {
18910450e9eSJacob Faibussowitsch   (void)comm;
19010450e9eSJacob Faibussowitsch   (void)line;
19110450e9eSJacob Faibussowitsch   (void)fun;
19210450e9eSJacob Faibussowitsch   (void)file;
19310450e9eSJacob Faibussowitsch   (void)p;
19410450e9eSJacob Faibussowitsch   (void)mess;
19510450e9eSJacob Faibussowitsch   (void)ctx;
196362febeeSStefano Zampini   return n;
197e93bc3c1Svictor }
198e93bc3c1Svictor 
199e5c89e4eSSatish Balay static char PetscErrorBaseMessage[1024];
200e5c89e4eSSatish Balay /*
201e5c89e4eSSatish Balay        The numerical values for these are defined in include/petscerror.h; any changes
202e5c89e4eSSatish Balay    there must also be made here
203e5c89e4eSSatish Balay */
204e5c89e4eSSatish Balay static const char *PetscErrorStrings[] = {
205e5c89e4eSSatish Balay   /*55 */ "Out of memory",
206e5c89e4eSSatish Balay   "No support for this operation for this object type",
207e5c89e4eSSatish Balay   "No support for this operation on this system",
208e5c89e4eSSatish Balay   /*58 */ "Operation done in wrong order",
209e5c89e4eSSatish Balay   /*59 */ "Signal received",
210e5c89e4eSSatish Balay   /*60 */ "Nonconforming object sizes",
211e5c89e4eSSatish Balay   "Argument aliasing not permitted",
212e5c89e4eSSatish Balay   "Invalid argument",
213e5c89e4eSSatish Balay   /*63 */ "Argument out of range",
214a17b96a8SKyle Gerard Felker   "Corrupt argument: https://petsc.org/release/faq/#valgrind",
215e5c89e4eSSatish Balay   "Unable to open file",
216e5c89e4eSSatish Balay   "Read from file failed",
217e5c89e4eSSatish Balay   "Write to file failed",
218e5c89e4eSSatish Balay   "Invalid pointer",
219e5c89e4eSSatish Balay   /*69 */ "Arguments must have same type",
220a8b45ee7SBarry Smith   /*70 */ "Attempt to use a pointer that does not point to a valid accessible location",
221a17b96a8SKyle Gerard Felker   /*71 */ "Zero pivot in LU factorization: https://petsc.org/release/faq/#zeropivot",
222e5c89e4eSSatish Balay   /*72 */ "Floating point exception",
223e5c89e4eSSatish Balay   /*73 */ "Object is in wrong state",
224e5c89e4eSSatish Balay   "Corrupted Petsc object",
225e5c89e4eSSatish Balay   "Arguments are incompatible",
226e5c89e4eSSatish Balay   "Error in external library",
227e5c89e4eSSatish Balay   /*77 */ "Petsc has generated inconsistent data",
228a17b96a8SKyle Gerard Felker   "Memory corruption: https://petsc.org/release/faq/#valgrind",
229e5c89e4eSSatish Balay   "Unexpected data in file",
230e5c89e4eSSatish Balay   /*80 */ "Arguments must have same communicators",
231a17b96a8SKyle Gerard Felker   /*81 */ "Zero pivot in Cholesky factorization: https://petsc.org/release/faq/#zeropivot",
232e5c89e4eSSatish Balay   "",
233e5c89e4eSSatish Balay   "",
234a17b96a8SKyle Gerard Felker   "Overflow in integer operation: https://petsc.org/release/faq/#64-bit-indices",
235e5c89e4eSSatish Balay   /*85 */ "Null argument, when expecting valid pointer",
236a17b96a8SKyle Gerard Felker   /*86 */ "Unknown type. Check for miss-spelling or missing package: https://petsc.org/release/install/install/#external-packages",
2373d96e996SBarry Smith   /*87 */ "MPI library at runtime is not compatible with MPI used at compile time",
2388cda6cd7SBarry Smith   /*88 */ "Error in system call",
239a17b96a8SKyle Gerard Felker   /*89 */ "Object Type not set: https://petsc.org/release/faq/#object-type-not-set",
24073260a9bSLisandro Dalcin   /*90 */ "",
24173260a9bSLisandro Dalcin   /*   */ "",
242a17b96a8SKyle Gerard Felker   /*92 */ "See https://petsc.org/release/overview/linear_solve_table/ for possible LU and Cholesky solvers",
243f560318cSPatrick Sanan   /*93 */ "You cannot overwrite this option since that will conflict with other previously set options",
244691b26d3SBarry Smith   /*94 */ "Example/application run with number of MPI ranks it does not support",
245691b26d3SBarry Smith   /*95 */ "Missing or incorrect user input",
246e57d7714SBarry Smith   /*96 */ "GPU resources unavailable",
247764761abSStefano Zampini   /*97 */ "GPU error",
248e809461dSJacob Faibussowitsch   /*98 */ "General MPI error",
249e809461dSJacob Faibussowitsch   /*99 */ "PetscError() incorrectly returned an error code of 0"};
250e5c89e4eSSatish Balay 
251e5c89e4eSSatish Balay /*@C
2523ba16761SJacob Faibussowitsch   PetscErrorMessage - Returns the text string associated with a PETSc error code.
253e5c89e4eSSatish Balay 
254cc4c1da9SBarry Smith   Not Collective, No Fortran Support
255e5c89e4eSSatish Balay 
256e5c89e4eSSatish Balay   Input Parameter:
257e5c89e4eSSatish Balay . errnum - the error code
258e5c89e4eSSatish Balay 
259d8d19677SJose E. Roman   Output Parameters:
2603ba16761SJacob Faibussowitsch + text     - the error message (`NULL` if not desired)
2613ba16761SJacob Faibussowitsch - specific - the specific error message that was set with `SETERRQ()` or
2623ba16761SJacob Faibussowitsch              `PetscError()`. (`NULL` if not desired)
263e5c89e4eSSatish Balay 
264e5c89e4eSSatish Balay   Level: developer
265e5c89e4eSSatish Balay 
2663ba16761SJacob Faibussowitsch .seealso: `PetscErrorCode`, `PetscPushErrorHandler()`, `PetscAttachDebuggerErrorHandler()`,
2673ba16761SJacob Faibussowitsch `PetscError()`, `SETERRQ()`, `PetscCall()` `PetscAbortErrorHandler()`,
2683ba16761SJacob Faibussowitsch `PetscTraceBackErrorHandler()`
269e5c89e4eSSatish Balay @*/
270cc4c1da9SBarry Smith PetscErrorCode PetscErrorMessage(PetscErrorCode errnum, const char *text[], char *specific[])
271d71ae5a4SJacob Faibussowitsch {
2723ba16761SJacob Faibussowitsch   PetscFunctionBegin;
2733ba16761SJacob Faibussowitsch   if (text) {
2743ba16761SJacob Faibussowitsch     if (errnum > PETSC_ERR_MIN_VALUE && errnum < PETSC_ERR_MAX_VALUE) {
275c35ec7c7SPierre Jolivet       size_t len;
276c35ec7c7SPierre Jolivet 
277c35ec7c7SPierre Jolivet       *text = PetscErrorStrings[errnum - PETSC_ERR_MIN_VALUE - 1];
278c35ec7c7SPierre Jolivet       PetscCall(PetscStrlen(*text, &len));
279c35ec7c7SPierre Jolivet       if (!len) *text = NULL;
2803ba16761SJacob Faibussowitsch     } else if (errnum == PETSC_ERR_BOOLEAN_MACRO_FAILURE) {
2813ba16761SJacob Faibussowitsch       /* this "error code" arises from failures in boolean macros, where the || operator is
2823ba16761SJacob Faibussowitsch          used to short-circuit the macro call in case of error. This has the side effect of
2833ba16761SJacob Faibussowitsch          "returning" either 0 (PETSC_SUCCESS) or 1 (PETSC_ERR_UNKNONWN):
284e5c89e4eSSatish Balay 
2853ba16761SJacob Faibussowitsch          #define PETSC_FOO(x) ((PetscErrorCode)(PetscBar(x) || PetscBaz(x)))
2863ba16761SJacob Faibussowitsch 
2873ba16761SJacob Faibussowitsch          If PetscBar() fails (returns nonzero) PetscBaz() is not executed but the result of
2883ba16761SJacob Faibussowitsch          this expression is boolean false, hence PETSC_ERR_UNNOWN
2893ba16761SJacob Faibussowitsch        */
2903ba16761SJacob Faibussowitsch       *text = "Error occurred in boolean shortcuit in macro";
2913ba16761SJacob Faibussowitsch     } else {
2923ba16761SJacob Faibussowitsch       *text = NULL;
2933ba16761SJacob Faibussowitsch     }
2943ba16761SJacob Faibussowitsch   }
295a297a907SKarl Rupp   if (specific) *specific = PetscErrorBaseMessage;
2963ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
297e5c89e4eSSatish Balay }
298e5c89e4eSSatish Balay 
299984a1229SMatthew G. Knepley #if defined(PETSC_CLANGUAGE_CXX)
300984a1229SMatthew G. Knepley   /* C++ exceptions are formally not allowed to propagate through extern "C" code. In practice, far too much software
30115229ffcSPierre Jolivet  * would be broken if implementations did not handle it in some common cases. However, keep in mind
302984a1229SMatthew G. Knepley  *
303984a1229SMatthew G. Knepley  *   Rule 62. Don't allow exceptions to propagate across module boundaries
304984a1229SMatthew G. Knepley  *
305984a1229SMatthew G. Knepley  * in "C++ Coding Standards" by Sutter and Alexandrescu. (This accounts for part of the ongoing C++ binary interface
306984a1229SMatthew G. Knepley  * instability.) Having PETSc raise errors as C++ exceptions was probably misguided and should eventually be removed.
307984a1229SMatthew G. Knepley  *
308984a1229SMatthew G. Knepley  * Here is the problem: You have a C++ function call a PETSc function, and you would like to maintain the error message
309984a1229SMatthew G. Knepley  * and stack information from the PETSc error. You could make everyone write exactly this code in their C++, but that
310984a1229SMatthew G. Knepley  * seems crazy to me.
311984a1229SMatthew G. Knepley  */
312984a1229SMatthew G. Knepley   #include <sstream>
3134c94c282SMatthew G. Knepley   #include <stdexcept>
314d71ae5a4SJacob Faibussowitsch static void PetscCxxErrorThrow()
315d71ae5a4SJacob Faibussowitsch {
316984a1229SMatthew G. Knepley   const char *str;
317984a1229SMatthew G. Knepley   if (eh && eh->ctx) {
318984a1229SMatthew G. Knepley     std::ostringstream *msg;
319984a1229SMatthew G. Knepley     msg = (std::ostringstream *)eh->ctx;
320984a1229SMatthew G. Knepley     str = msg->str().c_str();
321984a1229SMatthew G. Knepley   } else str = "Error detected in C PETSc";
322984a1229SMatthew G. Knepley 
323984a1229SMatthew G. Knepley   throw std::runtime_error(str);
324984a1229SMatthew G. Knepley }
325984a1229SMatthew G. Knepley #endif
326984a1229SMatthew G. Knepley 
327e5c89e4eSSatish Balay /*@C
3286e25c4a1SBarry Smith   PetscError - Routine that is called when an error has been detected, usually called through the macro `SETERRQ`(`PETSC_COMM_SELF`,)` or by `PetscCall()`.
329e5c89e4eSSatish Balay 
330811af0c4SBarry Smith   Collective
331e5c89e4eSSatish Balay 
332e5c89e4eSSatish Balay   Input Parameters:
3336e25c4a1SBarry Smith + comm - communicator over which error occurred.  ALL MPI processes of this communicator MUST call this routine
3346e25c4a1SBarry Smith . line - the line number of the error (usually indicated by `__LINE__` in the calling routine)
3353de71b31SHong Zhang . func - the function name in which the error was detected
3366e25c4a1SBarry Smith . file - the file in which the error was detected (usually indicated by `__FILE__` in the calling routine)
337e5c89e4eSSatish Balay . n    - the generic error number
338811af0c4SBarry Smith . p    - `PETSC_ERROR_INITIAL` indicates the error was initially detected, `PETSC_ERROR_REPEAT` indicates this is a traceback from a previously detected error
339e5c89e4eSSatish Balay - mess - formatted message string - aka printf
340e5c89e4eSSatish Balay 
341811af0c4SBarry Smith   Options Database Keys:
3426e25c4a1SBarry Smith + -error_output_stdout - output the error messages to `stdout` instead of the default `stderr`
34345b666d6SBarry Smith - -error_output_none   - do not output the error messages
34445b666d6SBarry Smith 
345e5c89e4eSSatish Balay   Level: intermediate
346e5c89e4eSSatish Balay 
347e5c89e4eSSatish Balay   Notes:
348e809461dSJacob Faibussowitsch   PETSc error handling is done with error return codes. A non-zero return indicates an error
349e809461dSJacob Faibussowitsch   was detected. The return-value of this routine is what is ultimately returned by
350e809461dSJacob Faibussowitsch   `SETERRQ()`.
35145b666d6SBarry Smith 
3526e25c4a1SBarry Smith   Numerical errors (potential divide by zero, for example) are not managed by the
353e809461dSJacob Faibussowitsch   error return codes; they are managed via, for example, `KSPGetConvergedReason()` that
354e809461dSJacob Faibussowitsch   indicates if the solve was successful or not. The option `-ksp_error_if_not_converged`, for
355e809461dSJacob Faibussowitsch   example, turns numerical failures into hard errors managed via `PetscError()`.
35645b666d6SBarry Smith 
357e809461dSJacob Faibussowitsch   PETSc provides a rich supply of error handlers, see the list below, and users can also
358e809461dSJacob Faibussowitsch   provide their own error handlers.
359811af0c4SBarry Smith 
360e809461dSJacob Faibussowitsch   If the user sets their own error handler (via `PetscPushErrorHandler()`) they may return any
361e809461dSJacob Faibussowitsch   arbitrary value from it, but are encouraged to return nonzero values. If the return value is
362e809461dSJacob Faibussowitsch   zero, `SETERRQ()` will ignore the value and return `PETSC_ERR_RETURN` (a nonzero value)
363e809461dSJacob Faibussowitsch   instead.
364e809461dSJacob Faibussowitsch 
365e809461dSJacob Faibussowitsch   Most users need not directly use this routine and the error handlers, but can instead use
366e809461dSJacob Faibussowitsch   the simplified interface `PetscCall()` or `SETERRQ()`.
367e5c89e4eSSatish Balay 
3686e25c4a1SBarry Smith   Fortran Note:
369e3081792SBarry Smith   This routine is used differently from Fortran
3706e25c4a1SBarry Smith .vb
3716e25c4a1SBarry Smith   PetscError(MPI_Comm comm, PetscErrorCode n, PetscErrorType p, char *message)
3726e25c4a1SBarry Smith .ve
373e3081792SBarry Smith 
3746e25c4a1SBarry Smith   Developer Note:
375811af0c4SBarry Smith   Since this is called after an error condition it should not be calling any error handlers (currently it ignores any error codes)
3763bf036e2SBarry 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
3773bf036e2SBarry Smith   but this annoying.
3783bf036e2SBarry Smith 
379db781477SPatrick Sanan .seealso: `PetscErrorCode`, `PetscPushErrorHandler()`, `PetscPopErrorHandler()`, `PetscTraceBackErrorHandler()`, `PetscAbortErrorHandler()`, `PetscMPIAbortErrorHandler()`,
380db781477SPatrick Sanan           `PetscReturnErrorHandler()`, `PetscAttachDebuggerErrorHandler()`, `PetscEmacsClientErrorHandler()`,
3816e25c4a1SBarry Smith           `SETERRQ()`, `PetscCall()`, `CHKMEMQ`, `PetscErrorMessage()`, `PETSCABORT()`, `PetscErrorType`, `PETSC_ERROR_INITIAL`, `PETSC_ERROR_REPEAT`
382e5c89e4eSSatish Balay @*/
383d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscError(MPI_Comm comm, int line, const char *func, const char *file, PetscErrorCode n, PetscErrorType p, const char *mess, ...)
384d71ae5a4SJacob Faibussowitsch {
385e5c89e4eSSatish Balay   va_list        Argp;
386c9a19010SBarry Smith   size_t         fullLength;
38702c9f0b5SLisandro Dalcin   char           buf[2048], *lbuf = NULL;
388fbfcfee5SBarry Smith   PetscBool      ismain;
3893bf036e2SBarry Smith   PetscErrorCode ierr;
390e5c89e4eSSatish Balay 
39139a651e2SJacob Faibussowitsch   if (!PetscErrorHandlingInitialized) return n;
3920d577829SBarry Smith   if (comm == MPI_COMM_NULL) comm = PETSC_COMM_SELF;
393e5c89e4eSSatish Balay 
394e5c89e4eSSatish Balay   /* Compose the message evaluating the print format */
395e5c89e4eSSatish Balay   if (mess) {
396e5c89e4eSSatish Balay     va_start(Argp, mess);
397*dd460d27SBarry Smith     (void)PetscVSNPrintf(buf, 2048, mess, &fullLength, Argp);
398e5c89e4eSSatish Balay     va_end(Argp);
399e5c89e4eSSatish Balay     lbuf = buf;
400*dd460d27SBarry Smith     if (p == PETSC_ERROR_INITIAL) (void)PetscStrncpy(PetscErrorBaseMessage, lbuf, sizeof(PetscErrorBaseMessage));
401e5c89e4eSSatish Balay   }
402e5c89e4eSSatish Balay 
403*dd460d27SBarry Smith   if (p == PETSC_ERROR_INITIAL && n != PETSC_ERR_MEMC) (void)PetscMallocValidate(__LINE__, PETSC_FUNCTION_NAME, __FILE__);
4044ed0ab5bSBarry Smith 
40502c9f0b5SLisandro Dalcin   if (!eh) ierr = PetscTraceBackErrorHandler(comm, line, func, file, n, p, lbuf, NULL);
406efca3c55SSatish Balay   else ierr = (*eh->handler)(comm, line, func, file, n, p, lbuf, eh->ctx);
407362febeeSStefano Zampini   PetscStackClearTop;
408e5c89e4eSSatish Balay 
409e5c89e4eSSatish Balay   /*
410989c49a2SBarry Smith       If this is called from the main() routine we abort the program.
411989c49a2SBarry Smith       We cannot just return because them some MPI processes may continue to attempt to run
412989c49a2SBarry Smith       while this process simply exits.
413e5c89e4eSSatish Balay   */
41449c86fc7SBarry Smith   if (func) {
415*dd460d27SBarry Smith     (void)PetscStrncmp(func, "main", 4, &ismain);
4167233276eSBarry Smith     if (ismain) {
417*dd460d27SBarry Smith       if (petscwaitonerrorflg) (void)PetscSleep(1000);
418660278c0SBarry Smith       PETSCABORT(comm, ierr);
4197233276eSBarry Smith     }
42049c86fc7SBarry Smith   }
4212c280183SJed Brown #if defined(PETSC_CLANGUAGE_CXX)
422ad540459SPierre Jolivet   if (p == PETSC_ERROR_IN_CXX) PetscCxxErrorThrow();
423d736bfebSBarry Smith #endif
424362febeeSStefano Zampini   return ierr;
425e5c89e4eSSatish Balay }
426e5c89e4eSSatish Balay 
427ffeef943SBarry Smith /*@
428e5c89e4eSSatish Balay   PetscIntView - Prints an array of integers; useful for debugging.
429e5c89e4eSSatish Balay 
430c3339decSBarry Smith   Collective
431e5c89e4eSSatish Balay 
432e5c89e4eSSatish Balay   Input Parameters:
433e5c89e4eSSatish Balay + N      - number of integers in array
434e5c89e4eSSatish Balay . idx    - array of integers
435811af0c4SBarry Smith - viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
436e5c89e4eSSatish Balay 
437e5c89e4eSSatish Balay   Level: intermediate
438e5c89e4eSSatish Balay 
439811af0c4SBarry Smith   Note:
440a3b724e8SBarry Smith   This may be called from within the debugger, passing 0 as the viewer
441300a7f5bSBarry Smith 
4426e25c4a1SBarry Smith   Developer Note:
4436e25c4a1SBarry Smith   `idx` cannot be const because may be passed to binary viewer where temporary byte swapping may be done
444811af0c4SBarry Smith 
445811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscRealView()`
446e5c89e4eSSatish Balay @*/
447d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscIntView(PetscInt N, const PetscInt idx[], PetscViewer viewer)
448d71ae5a4SJacob Faibussowitsch {
449ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
450e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 20, p = N % 20;
451ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
452e5c89e4eSSatish Balay   MPI_Comm    comm;
453e5c89e4eSSatish Balay 
454e5c89e4eSSatish Balay   PetscFunctionBegin;
455e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
4560700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
4574f572ea9SToby Isaac   if (N) PetscAssertPointer(idx, 2);
4589566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
4599566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
4609566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
461e5c89e4eSSatish Balay 
4629566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
4639566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
464e5c89e4eSSatish Balay   if (iascii) {
4659566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
466e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
467ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4689566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * i));
469ca0c3be5SJacob Faibussowitsch       } else {
4709566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * i));
471ca0c3be5SJacob Faibussowitsch       }
47248a46eb9SPierre Jolivet       for (j = 0; j < 20; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[i * 20 + j]));
4739566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
474e5c89e4eSSatish Balay     }
475e5c89e4eSSatish Balay     if (p) {
476ca0c3be5SJacob Faibussowitsch       if (size > 1) {
4779566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %" PetscInt_FMT ":", rank, 20 * n));
478ca0c3be5SJacob Faibussowitsch       } else {
4799566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%" PetscInt_FMT ":", 20 * n));
480ca0c3be5SJacob Faibussowitsch       }
4819566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %" PetscInt_FMT, idx[20 * n + i]));
4829566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
483e5c89e4eSSatish Balay     }
4849566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
4859566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
4866805f65bSBarry Smith   } else if (isbinary) {
487ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
488e5c89e4eSSatish Balay     PetscInt    *array;
489783b601eSJed Brown 
4909566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
491e5c89e4eSSatish Balay 
492e5c89e4eSSatish Balay     if (size > 1) {
493e5c89e4eSSatish Balay       if (rank) {
4949566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
4959566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, NULL, NULL, NULL, MPIU_INT, 0, comm));
496e5c89e4eSSatish Balay       } else {
4979566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
4989566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
499e5c89e4eSSatish Balay         Ntotal = sizes[0];
5009566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
501e5c89e4eSSatish Balay         displs[0] = 0;
502e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
503e5c89e4eSSatish Balay           Ntotal += sizes[i];
504e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
505e5c89e4eSSatish Balay         }
5069566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
5079566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_INT, array, sizes, displs, MPIU_INT, 0, comm));
5089566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_INT));
5099566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
5109566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
5119566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
512e5c89e4eSSatish Balay       }
513e5c89e4eSSatish Balay     } else {
5149566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, idx, N, PETSC_INT));
515e5c89e4eSSatish Balay     }
516e5c89e4eSSatish Balay   } else {
517e5c89e4eSSatish Balay     const char *tname;
5189566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
51998921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
520e5c89e4eSSatish Balay   }
5213ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
522e5c89e4eSSatish Balay }
523e5c89e4eSSatish Balay 
524ffeef943SBarry Smith /*@
525e5c89e4eSSatish Balay   PetscRealView - Prints an array of doubles; useful for debugging.
526e5c89e4eSSatish Balay 
527c3339decSBarry Smith   Collective
528e5c89e4eSSatish Balay 
529e5c89e4eSSatish Balay   Input Parameters:
530811af0c4SBarry Smith + N      - number of `PetscReal` in array
531811af0c4SBarry Smith . idx    - array of `PetscReal`
532811af0c4SBarry Smith - viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
533e5c89e4eSSatish Balay 
534e5c89e4eSSatish Balay   Level: intermediate
535e5c89e4eSSatish Balay 
536811af0c4SBarry Smith   Note:
537a3b724e8SBarry Smith   This may be called from within the debugger, passing 0 as the viewer
538300a7f5bSBarry Smith 
5396e25c4a1SBarry Smith   Developer Note:
5406e25c4a1SBarry Smith   `idx` cannot be const because may be passed to binary viewer where temporary byte swapping may be done
541811af0c4SBarry Smith 
542811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`
543e5c89e4eSSatish Balay @*/
544d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscRealView(PetscInt N, const PetscReal idx[], PetscViewer viewer)
545d71ae5a4SJacob Faibussowitsch {
546ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
547e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 5, p = N % 5;
548ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
549e5c89e4eSSatish Balay   MPI_Comm    comm;
550e5c89e4eSSatish Balay 
551e5c89e4eSSatish Balay   PetscFunctionBegin;
552e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
5530700a824SBarry Smith   PetscValidHeaderSpecific(viewer, PETSC_VIEWER_CLASSID, 3);
5544f572ea9SToby Isaac   PetscAssertPointer(idx, 2);
5559566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
5569566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
5579566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
558e5c89e4eSSatish Balay 
5599566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
5609566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
561e5c89e4eSSatish Balay   if (iascii) {
5621a989b97SToby Isaac     PetscInt tab;
5631a989b97SToby Isaac 
5649566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
5659566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIGetTab(viewer, &tab));
566e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
5679566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
568ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5699566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * i));
570ca0c3be5SJacob Faibussowitsch       } else {
5719566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * i));
572ca0c3be5SJacob Faibussowitsch       }
5739566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
57448a46eb9SPierre Jolivet       for (j = 0; j < 5; j++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 5 + j]));
5759566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
576e5c89e4eSSatish Balay     }
577e5c89e4eSSatish Balay     if (p) {
5789566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, tab));
579ca0c3be5SJacob Faibussowitsch       if (size > 1) {
5809566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 5 * n));
581ca0c3be5SJacob Faibussowitsch       } else {
5829566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 5 * n));
583ca0c3be5SJacob Faibussowitsch       }
5849566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISetTab(viewer, 0));
5859566063dSJacob Faibussowitsch       for (i = 0; i < p; i++) PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[5 * n + i]));
5869566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
587e5c89e4eSSatish Balay     }
5889566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
5899566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIISetTab(viewer, tab));
5909566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
5916805f65bSBarry Smith   } else if (isbinary) {
592ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, *displs, Ntotal, NN;
593e5c89e4eSSatish Balay     PetscReal   *array;
594e5c89e4eSSatish Balay 
5959566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
596e5c89e4eSSatish Balay 
597e5c89e4eSSatish Balay     if (size > 1) {
598e5c89e4eSSatish Balay       if (rank) {
5999566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
6009566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, NULL, NULL, NULL, MPIU_REAL, 0, comm));
601e5c89e4eSSatish Balay       } else {
6029566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
6039566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
604e5c89e4eSSatish Balay         Ntotal = sizes[0];
6059566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
606e5c89e4eSSatish Balay         displs[0] = 0;
607e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
608e5c89e4eSSatish Balay           Ntotal += sizes[i];
609e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
610e5c89e4eSSatish Balay         }
6119566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
6129566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((PetscReal *)idx, NN, MPIU_REAL, array, sizes, displs, MPIU_REAL, 0, comm));
6139566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_REAL));
6149566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
6159566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
6169566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
617e5c89e4eSSatish Balay       }
618e5c89e4eSSatish Balay     } else {
6199566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_REAL));
620e5c89e4eSSatish Balay     }
621e5c89e4eSSatish Balay   } else {
622e5c89e4eSSatish Balay     const char *tname;
6239566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
62498921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
625e5c89e4eSSatish Balay   }
6263ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
627e5c89e4eSSatish Balay }
628e5c89e4eSSatish Balay 
629ffeef943SBarry Smith /*@
630811af0c4SBarry Smith   PetscScalarView - Prints an array of `PetscScalar`; useful for debugging.
631e5c89e4eSSatish Balay 
632c3339decSBarry Smith   Collective
633e5c89e4eSSatish Balay 
634e5c89e4eSSatish Balay   Input Parameters:
635e5c89e4eSSatish Balay + N      - number of scalars in array
636e5c89e4eSSatish Balay . idx    - array of scalars
637811af0c4SBarry Smith - viewer - location to print array, `PETSC_VIEWER_STDOUT_WORLD`, `PETSC_VIEWER_STDOUT_SELF` or 0
638e5c89e4eSSatish Balay 
639e5c89e4eSSatish Balay   Level: intermediate
640e5c89e4eSSatish Balay 
641811af0c4SBarry Smith   Note:
642a3b724e8SBarry Smith   This may be called from within the debugger, passing 0 as the viewer
643300a7f5bSBarry Smith 
6446e25c4a1SBarry Smith   Developer Note:
6456e25c4a1SBarry Smith   `idx` cannot be const because may be passed to binary viewer where byte swapping may be done
646811af0c4SBarry Smith 
647811af0c4SBarry Smith .seealso: `PetscViewer`, `PetscIntView()`, `PetscRealView()`
648e5c89e4eSSatish Balay @*/
649d71ae5a4SJacob Faibussowitsch PetscErrorCode PetscScalarView(PetscInt N, const PetscScalar idx[], PetscViewer viewer)
650d71ae5a4SJacob Faibussowitsch {
651ca0c3be5SJacob Faibussowitsch   PetscMPIInt rank, size;
652e5c89e4eSSatish Balay   PetscInt    j, i, n = N / 3, p = N % 3;
653ace3abfcSBarry Smith   PetscBool   iascii, isbinary;
654e5c89e4eSSatish Balay   MPI_Comm    comm;
655e5c89e4eSSatish Balay 
656e5c89e4eSSatish Balay   PetscFunctionBegin;
657e5c89e4eSSatish Balay   if (!viewer) viewer = PETSC_VIEWER_STDOUT_SELF;
658e5c89e4eSSatish Balay   PetscValidHeader(viewer, 3);
6594f572ea9SToby Isaac   if (N) PetscAssertPointer(idx, 2);
6609566063dSJacob Faibussowitsch   PetscCall(PetscObjectGetComm((PetscObject)viewer, &comm));
6619566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_size(comm, &size));
6629566063dSJacob Faibussowitsch   PetscCallMPI(MPI_Comm_rank(comm, &rank));
663e5c89e4eSSatish Balay 
6649566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERASCII, &iascii));
6659566063dSJacob Faibussowitsch   PetscCall(PetscObjectTypeCompare((PetscObject)viewer, PETSCVIEWERBINARY, &isbinary));
666e5c89e4eSSatish Balay   if (iascii) {
6679566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPushSynchronized(viewer));
668e5c89e4eSSatish Balay     for (i = 0; i < n; i++) {
669ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6709566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * i));
671ca0c3be5SJacob Faibussowitsch       } else {
6729566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * i));
673ca0c3be5SJacob Faibussowitsch       }
674e5c89e4eSSatish Balay       for (j = 0; j < 3; j++) {
675e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6769566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[i * 3 + j]), (double)PetscImaginaryPart(idx[i * 3 + j])));
677e5c89e4eSSatish Balay #else
6789566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[i * 3 + j]));
679e5c89e4eSSatish Balay #endif
680e5c89e4eSSatish Balay       }
6819566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
682e5c89e4eSSatish Balay     }
683e5c89e4eSSatish Balay     if (p) {
684ca0c3be5SJacob Faibussowitsch       if (size > 1) {
6859566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "[%d] %2" PetscInt_FMT ":", rank, 3 * n));
686ca0c3be5SJacob Faibussowitsch       } else {
6879566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "%2" PetscInt_FMT ":", 3 * n));
688ca0c3be5SJacob Faibussowitsch       }
689e5c89e4eSSatish Balay       for (i = 0; i < p; i++) {
690e5c89e4eSSatish Balay #if defined(PETSC_USE_COMPLEX)
6919566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " (%12.4e,%12.4e)", (double)PetscRealPart(idx[n * 3 + i]), (double)PetscImaginaryPart(idx[n * 3 + i])));
692e5c89e4eSSatish Balay #else
6939566063dSJacob Faibussowitsch         PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, " %12.4e", (double)idx[3 * n + i]));
694e5c89e4eSSatish Balay #endif
695e5c89e4eSSatish Balay       }
6969566063dSJacob Faibussowitsch       PetscCall(PetscViewerASCIISynchronizedPrintf(viewer, "\n"));
697e5c89e4eSSatish Balay     }
6989566063dSJacob Faibussowitsch     PetscCall(PetscViewerFlush(viewer));
6999566063dSJacob Faibussowitsch     PetscCall(PetscViewerASCIIPopSynchronized(viewer));
7006805f65bSBarry Smith   } else if (isbinary) {
701ca0c3be5SJacob Faibussowitsch     PetscMPIInt *sizes, Ntotal, *displs, NN;
702e5c89e4eSSatish Balay     PetscScalar *array;
703e5c89e4eSSatish Balay 
7049566063dSJacob Faibussowitsch     PetscCall(PetscMPIIntCast(N, &NN));
705e5c89e4eSSatish Balay 
706e5c89e4eSSatish Balay     if (size > 1) {
707e5c89e4eSSatish Balay       if (rank) {
7089566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, NULL, 0, MPI_INT, 0, comm));
7099566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, NULL, NULL, NULL, MPIU_SCALAR, 0, comm));
710e5c89e4eSSatish Balay       } else {
7119566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &sizes));
7129566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gather(&NN, 1, MPI_INT, sizes, 1, MPI_INT, 0, comm));
713e5c89e4eSSatish Balay         Ntotal = sizes[0];
7149566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(size, &displs));
715e5c89e4eSSatish Balay         displs[0] = 0;
716e5c89e4eSSatish Balay         for (i = 1; i < size; i++) {
717e5c89e4eSSatish Balay           Ntotal += sizes[i];
718e5c89e4eSSatish Balay           displs[i] = displs[i - 1] + sizes[i - 1];
719e5c89e4eSSatish Balay         }
7209566063dSJacob Faibussowitsch         PetscCall(PetscMalloc1(Ntotal, &array));
7219566063dSJacob Faibussowitsch         PetscCallMPI(MPI_Gatherv((void *)idx, NN, MPIU_SCALAR, array, sizes, displs, MPIU_SCALAR, 0, comm));
7229566063dSJacob Faibussowitsch         PetscCall(PetscViewerBinaryWrite(viewer, array, Ntotal, PETSC_SCALAR));
7239566063dSJacob Faibussowitsch         PetscCall(PetscFree(sizes));
7249566063dSJacob Faibussowitsch         PetscCall(PetscFree(displs));
7259566063dSJacob Faibussowitsch         PetscCall(PetscFree(array));
726e5c89e4eSSatish Balay       }
727e5c89e4eSSatish Balay     } else {
7289566063dSJacob Faibussowitsch       PetscCall(PetscViewerBinaryWrite(viewer, (void *)idx, N, PETSC_SCALAR));
729e5c89e4eSSatish Balay     }
730e5c89e4eSSatish Balay   } else {
731e5c89e4eSSatish Balay     const char *tname;
7329566063dSJacob Faibussowitsch     PetscCall(PetscObjectGetName((PetscObject)viewer, &tname));
73398921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF, PETSC_ERR_SUP, "Cannot handle that PetscViewer of type %s", tname);
734e5c89e4eSSatish Balay   }
7353ba16761SJacob Faibussowitsch   PetscFunctionReturn(PETSC_SUCCESS);
736e5c89e4eSSatish Balay }
737e5c89e4eSSatish Balay 
738e22e20c5SJunchao Zhang #if defined(PETSC_HAVE_CUDA)
7390e6b6b59SJacob Faibussowitsch   #include <petscdevice_cuda.h>
740d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUBLASGetErrorName(cublasStatus_t status)
741d71ae5a4SJacob Faibussowitsch {
742e22e20c5SJunchao Zhang   switch (status) {
743e22e20c5SJunchao Zhang   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
744d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_SUCCESS:
745d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_SUCCESS";
746d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_INITIALIZED:
747d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_INITIALIZED";
748d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ALLOC_FAILED:
749d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ALLOC_FAILED";
750d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INVALID_VALUE:
751d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INVALID_VALUE";
752d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_ARCH_MISMATCH:
753d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_ARCH_MISMATCH";
754d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_MAPPING_ERROR:
755d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_MAPPING_ERROR";
756d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_EXECUTION_FAILED:
757d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_EXECUTION_FAILED";
758d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_INTERNAL_ERROR:
759d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_INTERNAL_ERROR";
760d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_NOT_SUPPORTED:
761d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_NOT_SUPPORTED";
762d71ae5a4SJacob Faibussowitsch   case CUBLAS_STATUS_LICENSE_ERROR:
763d71ae5a4SJacob Faibussowitsch     return "CUBLAS_STATUS_LICENSE_ERROR";
764e22e20c5SJunchao Zhang   #endif
765d71ae5a4SJacob Faibussowitsch   default:
766d71ae5a4SJacob Faibussowitsch     return "unknown error";
767e22e20c5SJunchao Zhang   }
768e22e20c5SJunchao Zhang }
769d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUSolverGetErrorName(cusolverStatus_t status)
770d71ae5a4SJacob Faibussowitsch {
771a4b895e1SBarry Smith   switch (status) {
772a4b895e1SBarry Smith   #if (CUDART_VERSION >= 8000) /* At least CUDA 8.0 of Sep. 2016 had these */
773d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_SUCCESS:
774d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_SUCCESS";
775d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_INITIALIZED:
776d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_INITIALIZED";
777d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_VALUE:
778d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_VALUE";
779d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ARCH_MISMATCH:
780d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ARCH_MISMATCH";
781d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INTERNAL_ERROR:
782d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INTERNAL_ERROR";
783030f984aSJacob Faibussowitsch     #if (CUDART_VERSION >= 9000) /* CUDA 9.0 had these defined on June 2021 */
784d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ALLOC_FAILED:
785d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ALLOC_FAILED";
786d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MAPPING_ERROR:
787d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MAPPING_ERROR";
788d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_EXECUTION_FAILED:
789d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_EXECUTION_FAILED";
790d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
791d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
792d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_NOT_SUPPORTED:
793d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_NOT_SUPPORTED ";
794d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_ZERO_PIVOT:
795d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_ZERO_PIVOT";
796d71ae5a4SJacob Faibussowitsch   case CUSOLVER_STATUS_INVALID_LICENSE:
797d71ae5a4SJacob Faibussowitsch     return "CUSOLVER_STATUS_INVALID_LICENSE";
798a4b895e1SBarry Smith     #endif
799030f984aSJacob Faibussowitsch   #endif
800d71ae5a4SJacob Faibussowitsch   default:
801d71ae5a4SJacob Faibussowitsch     return "unknown error";
802030f984aSJacob Faibussowitsch   }
803030f984aSJacob Faibussowitsch }
804d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscCUFFTGetErrorName(cufftResult result)
805d71ae5a4SJacob Faibussowitsch {
806030f984aSJacob Faibussowitsch   switch (result) {
807d71ae5a4SJacob Faibussowitsch   case CUFFT_SUCCESS:
808d71ae5a4SJacob Faibussowitsch     return "CUFFT_SUCCESS";
809d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_PLAN:
810d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_PLAN";
811d71ae5a4SJacob Faibussowitsch   case CUFFT_ALLOC_FAILED:
812d71ae5a4SJacob Faibussowitsch     return "CUFFT_ALLOC_FAILED";
813d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_TYPE:
814d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_TYPE";
815d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_VALUE:
816d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_VALUE";
817d71ae5a4SJacob Faibussowitsch   case CUFFT_INTERNAL_ERROR:
818d71ae5a4SJacob Faibussowitsch     return "CUFFT_INTERNAL_ERROR";
819d71ae5a4SJacob Faibussowitsch   case CUFFT_EXEC_FAILED:
820d71ae5a4SJacob Faibussowitsch     return "CUFFT_EXEC_FAILED";
821d71ae5a4SJacob Faibussowitsch   case CUFFT_SETUP_FAILED:
822d71ae5a4SJacob Faibussowitsch     return "CUFFT_SETUP_FAILED";
823d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_SIZE:
824d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_SIZE";
825d71ae5a4SJacob Faibussowitsch   case CUFFT_UNALIGNED_DATA:
826d71ae5a4SJacob Faibussowitsch     return "CUFFT_UNALIGNED_DATA";
827d71ae5a4SJacob Faibussowitsch   case CUFFT_INCOMPLETE_PARAMETER_LIST:
828d71ae5a4SJacob Faibussowitsch     return "CUFFT_INCOMPLETE_PARAMETER_LIST";
829d71ae5a4SJacob Faibussowitsch   case CUFFT_INVALID_DEVICE:
830d71ae5a4SJacob Faibussowitsch     return "CUFFT_INVALID_DEVICE";
831d71ae5a4SJacob Faibussowitsch   case CUFFT_PARSE_ERROR:
832d71ae5a4SJacob Faibussowitsch     return "CUFFT_PARSE_ERROR";
833d71ae5a4SJacob Faibussowitsch   case CUFFT_NO_WORKSPACE:
834d71ae5a4SJacob Faibussowitsch     return "CUFFT_NO_WORKSPACE";
835d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_IMPLEMENTED:
836d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_IMPLEMENTED";
837d71ae5a4SJacob Faibussowitsch   case CUFFT_LICENSE_ERROR:
838d71ae5a4SJacob Faibussowitsch     return "CUFFT_LICENSE_ERROR";
839d71ae5a4SJacob Faibussowitsch   case CUFFT_NOT_SUPPORTED:
840d71ae5a4SJacob Faibussowitsch     return "CUFFT_NOT_SUPPORTED";
841d71ae5a4SJacob Faibussowitsch   default:
842d71ae5a4SJacob Faibussowitsch     return "unknown error";
843a4b895e1SBarry Smith   }
844a4b895e1SBarry Smith }
845e22e20c5SJunchao Zhang #endif
84659af0bd3SScott Kruger 
84759af0bd3SScott Kruger #if defined(PETSC_HAVE_HIP)
8480e6b6b59SJacob Faibussowitsch   #include <petscdevice_hip.h>
849d71ae5a4SJacob Faibussowitsch PETSC_EXTERN const char *PetscHIPBLASGetErrorName(hipblasStatus_t status)
850d71ae5a4SJacob Faibussowitsch {
85159af0bd3SScott Kruger   switch (status) {
852d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_SUCCESS:
853d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_SUCCESS";
854d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_INITIALIZED:
855d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_INITIALIZED";
856d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ALLOC_FAILED:
857d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ALLOC_FAILED";
858d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INVALID_VALUE:
859d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INVALID_VALUE";
860d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_ARCH_MISMATCH:
861d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_ARCH_MISMATCH";
862d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_MAPPING_ERROR:
863d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_MAPPING_ERROR";
864d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_EXECUTION_FAILED:
865d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_EXECUTION_FAILED";
866d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_INTERNAL_ERROR:
867d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_INTERNAL_ERROR";
868d71ae5a4SJacob Faibussowitsch   case HIPBLAS_STATUS_NOT_SUPPORTED:
869d71ae5a4SJacob Faibussowitsch     return "HIPBLAS_STATUS_NOT_SUPPORTED";
870d71ae5a4SJacob Faibussowitsch   default:
871d71ae5a4SJacob Faibussowitsch     return "unknown error";
87259af0bd3SScott Kruger   }
87359af0bd3SScott Kruger }
87447d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSPARSEGetErrorName(hipsparseStatus_t status)
87547d993e7Ssuyashtn {
87647d993e7Ssuyashtn   switch (status) {
87747d993e7Ssuyashtn   case HIPSPARSE_STATUS_SUCCESS:
87847d993e7Ssuyashtn     return "HIPSPARSE_STATUS_SUCCESS";
87947d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_INITIALIZED:
88047d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_INITIALIZED";
88147d993e7Ssuyashtn   case HIPSPARSE_STATUS_ALLOC_FAILED:
88247d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ALLOC_FAILED";
88347d993e7Ssuyashtn   case HIPSPARSE_STATUS_INVALID_VALUE:
88447d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INVALID_VALUE";
88547d993e7Ssuyashtn   case HIPSPARSE_STATUS_ARCH_MISMATCH:
88647d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ARCH_MISMATCH";
88747d993e7Ssuyashtn   case HIPSPARSE_STATUS_MAPPING_ERROR:
88847d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MAPPING_ERROR";
88947d993e7Ssuyashtn   case HIPSPARSE_STATUS_EXECUTION_FAILED:
89047d993e7Ssuyashtn     return "HIPSPARSE_STATUS_EXECUTION_FAILED";
89147d993e7Ssuyashtn   case HIPSPARSE_STATUS_INTERNAL_ERROR:
89247d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INTERNAL_ERROR";
89347d993e7Ssuyashtn   case HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED:
89447d993e7Ssuyashtn     return "HIPSPARSE_STATUS_MATRIX_TYPE_NOT_SUPPORTED";
89547d993e7Ssuyashtn   case HIPSPARSE_STATUS_ZERO_PIVOT:
89647d993e7Ssuyashtn     return "HIPSPARSE_STATUS_ZERO_PIVOT";
89747d993e7Ssuyashtn   case HIPSPARSE_STATUS_NOT_SUPPORTED:
89847d993e7Ssuyashtn     return "HIPSPARSE_STATUS_NOT_SUPPORTED";
89947d993e7Ssuyashtn   case HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES:
90047d993e7Ssuyashtn     return "HIPSPARSE_STATUS_INSUFFICIENT_RESOURCES";
90147d993e7Ssuyashtn   default:
90247d993e7Ssuyashtn     return "unknown error";
90347d993e7Ssuyashtn   }
90447d993e7Ssuyashtn }
90547d993e7Ssuyashtn PETSC_EXTERN const char *PetscHIPSolverGetErrorName(hipsolverStatus_t status)
90647d993e7Ssuyashtn {
90747d993e7Ssuyashtn   switch (status) {
90847d993e7Ssuyashtn   case HIPSOLVER_STATUS_SUCCESS:
90947d993e7Ssuyashtn     return "HIPSOLVER_STATUS_SUCCESS";
91047d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_INITIALIZED:
91147d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_INITIALIZED";
91247d993e7Ssuyashtn   case HIPSOLVER_STATUS_ALLOC_FAILED:
91347d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ALLOC_FAILED";
91447d993e7Ssuyashtn   case HIPSOLVER_STATUS_MAPPING_ERROR:
91547d993e7Ssuyashtn     return "HIPSOLVER_STATUS_MAPPING_ERROR";
91647d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_VALUE:
91747d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_VALUE";
91847d993e7Ssuyashtn   case HIPSOLVER_STATUS_EXECUTION_FAILED:
91947d993e7Ssuyashtn     return "HIPSOLVER_STATUS_EXECUTION_FAILED";
92047d993e7Ssuyashtn   case HIPSOLVER_STATUS_INTERNAL_ERROR:
92147d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INTERNAL_ERROR";
92247d993e7Ssuyashtn   case HIPSOLVER_STATUS_NOT_SUPPORTED:
92347d993e7Ssuyashtn     return "HIPSOLVER_STATUS_NOT_SUPPORTED ";
92447d993e7Ssuyashtn   case HIPSOLVER_STATUS_ARCH_MISMATCH:
92547d993e7Ssuyashtn     return "HIPSOLVER_STATUS_ARCH_MISMATCH";
92647d993e7Ssuyashtn   case HIPSOLVER_STATUS_HANDLE_IS_NULLPTR:
92747d993e7Ssuyashtn     return "HIPSOLVER_STATUS_HANDLE_IS_NULLPTR";
92847d993e7Ssuyashtn   case HIPSOLVER_STATUS_INVALID_ENUM:
92947d993e7Ssuyashtn     return "HIPSOLVER_STATUS_INVALID_ENUM";
93047d993e7Ssuyashtn   case HIPSOLVER_STATUS_UNKNOWN:
93147d993e7Ssuyashtn   default:
93247d993e7Ssuyashtn     return "HIPSOLVER_STATUS_UNKNOWN";
93347d993e7Ssuyashtn   }
93447d993e7Ssuyashtn }
93559af0bd3SScott Kruger #endif
936db9cea48SBarry Smith 
937aec76313SJacob Faibussowitsch /*@C
938811af0c4SBarry Smith   PetscMPIErrorString - Given an MPI error code returns the `MPI_Error_string()` appropriately
939db9cea48SBarry Smith   formatted for displaying with the PETSc error handlers.
940db9cea48SBarry Smith 
941cc4c1da9SBarry Smith   Not Collective, No Fortran Support
942cc4c1da9SBarry Smith 
943a1fd7ae3SBarry Smith   Input Parameters:
944a1fd7ae3SBarry Smith + err  - the MPI error code
945a1fd7ae3SBarry Smith - slen - length of `string`, should be at least as large as `MPI_MAX_ERROR_STRING`
946db9cea48SBarry Smith 
947db9cea48SBarry Smith   Output Parameter:
948a1fd7ae3SBarry Smith . string - the MPI error message
949db9cea48SBarry Smith 
95049c86fc7SBarry Smith   Level: developer
95149c86fc7SBarry Smith 
952811af0c4SBarry Smith   Note:
953db9cea48SBarry Smith   Does not return an error code or do error handling because it may be called from inside an error handler
954db9cea48SBarry Smith 
95510450e9eSJacob Faibussowitsch .seealso: `PetscErrorCode` `PetscErrorMessage()`
956db9cea48SBarry Smith @*/
957a1fd7ae3SBarry Smith void PetscMPIErrorString(PetscMPIInt err, size_t slen, char *string)
958d71ae5a4SJacob Faibussowitsch {
959db9cea48SBarry Smith   char        errorstring[MPI_MAX_ERROR_STRING];
960a1fd7ae3SBarry Smith   PetscMPIInt len;
961a1fd7ae3SBarry Smith   size_t      j = 0;
962db9cea48SBarry Smith 
963db9cea48SBarry Smith   MPI_Error_string(err, (char *)errorstring, &len);
964a1fd7ae3SBarry Smith   for (PetscMPIInt i = 0; i < len && j < slen - 2; i++) {
965db9cea48SBarry Smith     string[j++] = errorstring[i];
966db9cea48SBarry Smith     if (errorstring[i] == '\n') {
967a1fd7ae3SBarry Smith       for (PetscMPIInt k = 0; k < 16 && j < slen - 2; k++) string[j++] = ' ';
968db9cea48SBarry Smith     }
969db9cea48SBarry Smith   }
970db9cea48SBarry Smith   string[j] = 0;
971db9cea48SBarry Smith }
972