xref: /petsc/src/sys/dll/dlimpl.c (revision 28b400f66ebc7ae0049166a2294dfcd3df27e64b)
1ebd79076SLisandro Dalcin /*
2ebd79076SLisandro Dalcin    Low-level routines for managing dynamic link libraries (DLLs).
3ebd79076SLisandro Dalcin */
4ebd79076SLisandro Dalcin 
5184ce1a2SMatthew G. Knepley #include <petscconf.h>
6184ce1a2SMatthew G. Knepley #if defined(PETSC__GNU_SOURCE)
7184ce1a2SMatthew G. Knepley   #if !defined(_GNU_SOURCE)
8184ce1a2SMatthew G. Knepley     #define _GNU_SOURCE 1
9184ce1a2SMatthew G. Knepley   #endif
10184ce1a2SMatthew G. Knepley #endif
11184ce1a2SMatthew G. Knepley 
12af0996ceSBarry Smith #include <petsc/private/petscimpl.h>
13ebd79076SLisandro Dalcin 
14ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_WINDOWS_H)
15ebd79076SLisandro Dalcin #include <windows.h>
16184ce1a2SMatthew G. Knepley #endif
17184ce1a2SMatthew G. Knepley #if defined(PETSC_HAVE_DLFCN_H)
18ebd79076SLisandro Dalcin #include <dlfcn.h>
19ebd79076SLisandro Dalcin #endif
20ebd79076SLisandro Dalcin 
21ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_WINDOWS_H)
22ebd79076SLisandro Dalcin typedef HMODULE dlhandle_t;
236ea75d68SLisandro Dalcin typedef FARPROC dlsymbol_t;
24ebd79076SLisandro Dalcin #elif defined(PETSC_HAVE_DLFCN_H)
25ebd79076SLisandro Dalcin typedef void* dlhandle_t;
266ea75d68SLisandro Dalcin typedef void* dlsymbol_t;
27ebd79076SLisandro Dalcin #else
28ebd79076SLisandro Dalcin typedef void* dlhandle_t;
296ea75d68SLisandro Dalcin typedef void* dlsymbol_t;
30ebd79076SLisandro Dalcin #endif
31ebd79076SLisandro Dalcin 
32ebd79076SLisandro Dalcin /*@C
33b235ab32SBarry Smith    PetscDLOpen - opens dynamic library
34b235ab32SBarry Smith 
35b235ab32SBarry Smith    Not Collective
36b235ab32SBarry Smith 
37b235ab32SBarry Smith    Input Parameters:
38b235ab32SBarry Smith +    name - name of library
39e4d177b5SBarry Smith -    mode - options on how to open library
40b235ab32SBarry Smith 
41b235ab32SBarry Smith    Output Parameter:
423222ab0cSSatish Balay .    handle - opaque pointer to be used with PetscDLSym()
43b235ab32SBarry Smith 
44b235ab32SBarry Smith    Level: developer
45b235ab32SBarry Smith 
4610699b91SBarry Smith .seealso: PetscDLClose(), PetscDLSym(), PetscDLAddr()
47ebd79076SLisandro Dalcin @*/
48e4d177b5SBarry Smith PetscErrorCode  PetscDLOpen(const char name[],PetscDLMode mode,PetscDLHandle *handle)
49ebd79076SLisandro Dalcin {
508e049581SJed Brown   PETSC_UNUSED int dlflags1,dlflags2; /* There are some preprocessor paths where these variables are set, but not used */
51ebd79076SLisandro Dalcin   dlhandle_t       dlhandle;
52ebd79076SLisandro Dalcin 
53ebd79076SLisandro Dalcin   PetscFunctionBegin;
54ebd79076SLisandro Dalcin   PetscValidCharPointer(name,1);
55ebd79076SLisandro Dalcin   PetscValidPointer(handle,3);
56ebd79076SLisandro Dalcin 
57be1c6ad7SLisandro Dalcin   dlflags1 = 0;
58be1c6ad7SLisandro Dalcin   dlflags2 = 0;
595673baf8SLisandro Dalcin   dlhandle = (dlhandle_t) 0;
60be1c6ad7SLisandro Dalcin   *handle  = (PetscDLHandle) 0;
61ebd79076SLisandro Dalcin 
625673baf8SLisandro Dalcin   /*
635673baf8SLisandro Dalcin      --- LoadLibrary ---
645673baf8SLisandro Dalcin   */
65be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_WINDOWS_H) && defined(PETSC_HAVE_LOADLIBRARY)
66be1c6ad7SLisandro Dalcin   dlhandle = LoadLibrary(name);
67ebd79076SLisandro Dalcin   if (!dlhandle) {
68ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_GETLASTERROR)
69ebd79076SLisandro Dalcin     PetscErrorCode ierr;
70ebd79076SLisandro Dalcin     DWORD          erc;
71b3bb0f5eSLisandro Dalcin     char           *buff = NULL;
72ebd79076SLisandro Dalcin     erc = GetLastError();
73ebd79076SLisandro Dalcin     FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_IGNORE_INSERTS,
740298fd71SBarry Smith                   NULL,erc,MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),(LPSTR)&buff,0,NULL);
75fbfcfee5SBarry Smith     ierr = PetscError(PETSC_COMM_SELF,__LINE__,PETSC_FUNCTION_NAME,__FILE__,PETSC_ERR_FILE_OPEN,PETSC_ERROR_REPEAT,
76be1c6ad7SLisandro Dalcin                       "Unable to open dynamic library:\n  %s\n  Error message from LoadLibrary() %s\n",name,buff);
77ebd79076SLisandro Dalcin     LocalFree(buff);
78ebd79076SLisandro Dalcin     PetscFunctionReturn(ierr);
79ebd79076SLisandro Dalcin #else
8098921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to open dynamic library:\n  %s\n  Error message from LoadLibrary() %s",name,"unavailable");
81ebd79076SLisandro Dalcin #endif
82ebd79076SLisandro Dalcin   }
835673baf8SLisandro Dalcin 
845673baf8SLisandro Dalcin   /*
855673baf8SLisandro Dalcin      --- dlopen ---
865673baf8SLisandro Dalcin   */
87a21658a3SLisandro Dalcin #elif defined(PETSC_HAVE_DLFCN_H) && defined(PETSC_HAVE_DLOPEN)
88ebd79076SLisandro Dalcin   /*
89ebd79076SLisandro Dalcin       Mode indicates symbols required by symbol loaded with dlsym()
90ebd79076SLisandro Dalcin      are only loaded when required (not all together) also indicates
91ebd79076SLisandro Dalcin      symbols required can be contained in other libraries also opened
92ebd79076SLisandro Dalcin      with dlopen()
93ebd79076SLisandro Dalcin   */
94be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_RTLD_LAZY)
95be1c6ad7SLisandro Dalcin   dlflags1 = RTLD_LAZY;
96be1c6ad7SLisandro Dalcin #endif
97ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_RTLD_NOW)
98a297a907SKarl Rupp   if (mode & PETSC_DL_NOW) dlflags1 = RTLD_NOW;
99ebd79076SLisandro Dalcin #endif
100ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_RTLD_GLOBAL)
101be1c6ad7SLisandro Dalcin   dlflags2 = RTLD_GLOBAL;
102ebd79076SLisandro Dalcin #endif
103b3bb0f5eSLisandro Dalcin #if defined(PETSC_HAVE_RTLD_LOCAL)
104a297a907SKarl Rupp   if (mode & PETSC_DL_LOCAL) dlflags2 = RTLD_LOCAL;
105b3bb0f5eSLisandro Dalcin #endif
106be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR)
107be1c6ad7SLisandro Dalcin   dlerror(); /* clear any previous error */
108be1c6ad7SLisandro Dalcin #endif
109be1c6ad7SLisandro Dalcin   dlhandle = dlopen(name,dlflags1|dlflags2);
110ebd79076SLisandro Dalcin   if (!dlhandle) {
111ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR)
112ebd79076SLisandro Dalcin     const char *errmsg = dlerror();
113ebd79076SLisandro Dalcin #else
114ebd79076SLisandro Dalcin     const char *errmsg = "unavailable";
115ebd79076SLisandro Dalcin #endif
11698921bdaSJacob Faibussowitsch     SETERRQ(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to open dynamic library:\n  %s\n  Error message from dlopen() %s",name,errmsg);
117ebd79076SLisandro Dalcin   }
1185673baf8SLisandro Dalcin   /*
1195673baf8SLisandro Dalcin      --- unimplemented ---
1205673baf8SLisandro Dalcin   */
121ebd79076SLisandro Dalcin #else
122e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS, "Cannot use dynamic libraries on this platform");
123ebd79076SLisandro Dalcin #endif
124ebd79076SLisandro Dalcin 
125ebd79076SLisandro Dalcin   *handle = (PetscDLHandle) dlhandle;
126ebd79076SLisandro Dalcin   PetscFunctionReturn(0);
127ebd79076SLisandro Dalcin }
128ebd79076SLisandro Dalcin 
129ebd79076SLisandro Dalcin /*@C
130b235ab32SBarry Smith    PetscDLClose -  closes a dynamic library
131b235ab32SBarry Smith 
132b235ab32SBarry Smith    Not Collective
133b235ab32SBarry Smith 
134b235ab32SBarry Smith   Input Parameter:
135b235ab32SBarry Smith .   handle - the handle for the library obtained with PetscDLOpen()
136b235ab32SBarry Smith 
137b235ab32SBarry Smith   Level: developer
13810699b91SBarry Smith 
13910699b91SBarry Smith .seealso: PetscDLOpen(), PetscDLSym(), PetscDLAddr()
140ebd79076SLisandro Dalcin @*/
1417087cfbeSBarry Smith PetscErrorCode  PetscDLClose(PetscDLHandle *handle)
142ebd79076SLisandro Dalcin {
143ebd79076SLisandro Dalcin   PetscFunctionBegin;
144ebd79076SLisandro Dalcin   PetscValidPointer(handle,1);
145ebd79076SLisandro Dalcin 
1465673baf8SLisandro Dalcin   /*
1475673baf8SLisandro Dalcin      --- FreeLibrary ---
1485673baf8SLisandro Dalcin   */
149ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_WINDOWS_H)
150be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_FREELIBRARY)
1518e049581SJed Brown   if (FreeLibrary((dlhandle_t)*handle) == 0) {
1525673baf8SLisandro Dalcin #if defined(PETSC_HAVE_GETLASTERROR)
153b3bb0f5eSLisandro Dalcin     char  *buff = NULL;
154a21658a3SLisandro Dalcin     DWORD erc   = GetLastError();
1550298fd71SBarry Smith     FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_IGNORE_INSERTS,NULL,erc,MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),(LPSTR)&buff,0,NULL);
1565673baf8SLisandro Dalcin     PetscErrorPrintf("Error closing dynamic library:\n  Error message from FreeLibrary() %s\n",buff);
1575673baf8SLisandro Dalcin     LocalFree(buff);
1585673baf8SLisandro Dalcin #else
1595673baf8SLisandro Dalcin     PetscErrorPrintf("Error closing dynamic library:\n  Error message from FreeLibrary() %s\n","unavailable");
1605673baf8SLisandro Dalcin #endif
1615673baf8SLisandro Dalcin   }
162be1c6ad7SLisandro Dalcin #endif /* !PETSC_HAVE_FREELIBRARY */
163ebd79076SLisandro Dalcin 
1645673baf8SLisandro Dalcin   /*
1655673baf8SLisandro Dalcin      --- dclose ---
1665673baf8SLisandro Dalcin   */
167ebd79076SLisandro Dalcin #elif defined(PETSC_HAVE_DLFCN_H)
168a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_DLCLOSE)
169be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR)
170be1c6ad7SLisandro Dalcin   dlerror(); /* clear any previous error */
171be1c6ad7SLisandro Dalcin #endif
1728e049581SJed Brown   if (dlclose((dlhandle_t)*handle) < 0) {
173ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR)
174ebd79076SLisandro Dalcin     const char *errmsg = dlerror();
175ebd79076SLisandro Dalcin #else
176ebd79076SLisandro Dalcin     const char *errmsg = "unavailable";
177ebd79076SLisandro Dalcin #endif
1785673baf8SLisandro Dalcin     PetscErrorPrintf("Error closing dynamic library:\n  Error message from dlclose() %s\n", errmsg);
179ebd79076SLisandro Dalcin   }
180a21658a3SLisandro Dalcin #endif /* !PETSC_HAVE_DLCLOSE */
1815673baf8SLisandro Dalcin 
1825673baf8SLisandro Dalcin   /*
1835673baf8SLisandro Dalcin      --- unimplemented ---
1845673baf8SLisandro Dalcin   */
185ebd79076SLisandro Dalcin #else
186e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS, "Cannot use dynamic libraries on this platform");
187ebd79076SLisandro Dalcin #endif
188ebd79076SLisandro Dalcin 
1890298fd71SBarry Smith   *handle = NULL;
190ebd79076SLisandro Dalcin   PetscFunctionReturn(0);
191ebd79076SLisandro Dalcin }
192ebd79076SLisandro Dalcin 
193ebd79076SLisandro Dalcin /*@C
194b235ab32SBarry Smith    PetscDLSym - finds a symbol in a dynamic library
195b235ab32SBarry Smith 
196b235ab32SBarry Smith    Not Collective
197b235ab32SBarry Smith 
198b235ab32SBarry Smith    Input Parameters:
1990298fd71SBarry Smith +   handle - obtained with PetscDLOpen() or NULL
200b235ab32SBarry Smith -   symbol - name of symbol
201b235ab32SBarry Smith 
202b235ab32SBarry Smith    Output Parameter:
2030298fd71SBarry Smith .   value - pointer to the function, NULL if not found
204b235ab32SBarry Smith 
205b235ab32SBarry Smith    Level: developer
206b235ab32SBarry Smith 
2077c62f5d3SDmitry Karpeev   Notes:
2080298fd71SBarry Smith    If handle is NULL, the symbol is looked for in the main executable's dynamic symbol table.
2097c62f5d3SDmitry Karpeev    In order to be dynamically loadable, the symbol has to be exported as such.  On many UNIX-like
2107c62f5d3SDmitry Karpeev    systems this requires platform-specific linker flags.
2117c62f5d3SDmitry Karpeev 
21210699b91SBarry Smith .seealso: PetscDLClose(), PetscDLOpen(), PetscDLAddr()
213ebd79076SLisandro Dalcin @*/
2147087cfbeSBarry Smith PetscErrorCode  PetscDLSym(PetscDLHandle handle,const char symbol[],void **value)
215ebd79076SLisandro Dalcin {
2168e049581SJed Brown   PETSC_UNUSED dlhandle_t dlhandle;
2176ea75d68SLisandro Dalcin   dlsymbol_t              dlsymbol;
218ebd79076SLisandro Dalcin 
219ebd79076SLisandro Dalcin   PetscValidCharPointer(symbol,2);
220ebd79076SLisandro Dalcin   PetscValidPointer(value,3);
221ebd79076SLisandro Dalcin 
222be1c6ad7SLisandro Dalcin   dlhandle = (dlhandle_t) 0;
2236ea75d68SLisandro Dalcin   dlsymbol = (dlsymbol_t) 0;
224a21658a3SLisandro Dalcin   *value   = (void*) 0;
225ebd79076SLisandro Dalcin 
2265673baf8SLisandro Dalcin   /*
2275673baf8SLisandro Dalcin      --- GetProcAddress ---
2285673baf8SLisandro Dalcin   */
229a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_WINDOWS_H)
230a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_GETPROCADDRESS)
231e4d177b5SBarry Smith   if (handle) dlhandle = (dlhandle_t) handle;
232e4d177b5SBarry Smith   else dlhandle = (dlhandle_t) GetCurrentProcess();
2336ea75d68SLisandro Dalcin   dlsymbol = (dlsymbol_t) GetProcAddress(dlhandle,symbol);
234a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_SETLASTERROR)
235a21658a3SLisandro Dalcin   SetLastError((DWORD)0); /* clear any previous error */
236a21658a3SLisandro Dalcin #endif
237a21658a3SLisandro Dalcin #endif /* !PETSC_HAVE_GETPROCADDRESS */
2385673baf8SLisandro Dalcin 
2395673baf8SLisandro Dalcin   /*
2405673baf8SLisandro Dalcin      --- dlsym ---
2415673baf8SLisandro Dalcin   */
242ebd79076SLisandro Dalcin #elif defined(PETSC_HAVE_DLFCN_H)
243a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_DLSYM)
244a297a907SKarl Rupp   if (handle) dlhandle = (dlhandle_t) handle;
245a297a907SKarl Rupp   else {
2467c62f5d3SDmitry Karpeev 
2475617e356SJunchao Zhang #if defined(PETSC_HAVE_DLOPEN)
2487c62f5d3SDmitry Karpeev     /* Attempt to retrieve the main executable's dlhandle. */
2497c62f5d3SDmitry Karpeev     { int dlflags1 = 0, dlflags2 = 0;
2507c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_RTLD_LAZY)
2517c62f5d3SDmitry Karpeev       dlflags1 = RTLD_LAZY;
2527c62f5d3SDmitry Karpeev #endif
2537c62f5d3SDmitry Karpeev       if (!dlflags1) {
2547c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_RTLD_NOW)
2557c62f5d3SDmitry Karpeev         dlflags1 = RTLD_NOW;
2567c62f5d3SDmitry Karpeev #endif
2577c62f5d3SDmitry Karpeev       }
2587c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_RTLD_LOCAL)
2597c62f5d3SDmitry Karpeev       dlflags2 = RTLD_LOCAL;
2607c62f5d3SDmitry Karpeev #endif
2617c62f5d3SDmitry Karpeev       if (!dlflags2) {
2627c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_RTLD_GLOBAL)
2637c62f5d3SDmitry Karpeev         dlflags2 = RTLD_GLOBAL;
2647c62f5d3SDmitry Karpeev #endif
2657c62f5d3SDmitry Karpeev       }
2667c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_DLERROR)
267cae2f346SBarry Smith       if (!(PETSC_RUNNING_ON_VALGRIND)) {
2687bb14e67SBarry Smith         dlerror(); /* clear any previous error; valgrind does not like this */
2697bb14e67SBarry Smith       }
2707bb14e67SBarry Smith #endif
2717c62f5d3SDmitry Karpeev       /* Attempt to open the main executable as a dynamic library. */
2727d421530SBarry Smith #if defined(PETSC_HAVE_RTDL_DEFAULT)
2737d421530SBarry Smith       dlhandle = RTLD_DEFAULT;
2747d421530SBarry Smith #else
27502c9f0b5SLisandro Dalcin       dlhandle = dlopen(NULL, dlflags1|dlflags2);
2767c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_DLERROR)
2774956914eSSatish Balay       { const char *e = (const char*) dlerror();
278*28b400f6SJacob Faibussowitsch         PetscCheck(!e,PETSC_COMM_SELF, PETSC_ERR_ARG_WRONG, "Error opening main executable as a dynamic library:\n  Error message from dlopen(): '%s'", e);
2797c62f5d3SDmitry Karpeev       }
2807c62f5d3SDmitry Karpeev #endif
2817d421530SBarry Smith #endif
282be1c6ad7SLisandro Dalcin     }
283be1c6ad7SLisandro Dalcin #endif
284aa2d57e9SJed Brown #endif /* PETSC_HAVE_DLOPEN && PETSC_HAVE_DYNAMIC_LIBRARIES */
2854956914eSSatish Balay   }
286be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR)
287be1c6ad7SLisandro Dalcin   dlerror(); /* clear any previous error */
288be1c6ad7SLisandro Dalcin #endif
2896ea75d68SLisandro Dalcin   dlsymbol = (dlsymbol_t) dlsym(dlhandle,symbol);
2905673baf8SLisandro Dalcin   /*
2915673baf8SLisandro Dalcin      --- unimplemented ---
2925673baf8SLisandro Dalcin   */
293ebd79076SLisandro Dalcin #else
294e32f2f54SBarry Smith   SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS, "Cannot use dynamic libraries on this platform");
295ebd79076SLisandro Dalcin #endif
296ebd79076SLisandro Dalcin 
2976ea75d68SLisandro Dalcin   *value = *((void**)&dlsymbol);
298ebd79076SLisandro Dalcin 
2992d53ad75SBarry Smith #if defined(PETSC_SERIALIZE_FUNCTIONS)
3005f80ce2aSJacob Faibussowitsch   if (*value) CHKERRQ(PetscFPTAdd(*value,symbol));
3012d53ad75SBarry Smith #endif
3022d53ad75SBarry Smith   return(0);
303ebd79076SLisandro Dalcin }
304184ce1a2SMatthew G. Knepley 
305184ce1a2SMatthew G. Knepley /*@C
306184ce1a2SMatthew G. Knepley   PetscDLAddr - find the name of a symbol in a dynamic library
307184ce1a2SMatthew G. Knepley 
308184ce1a2SMatthew G. Knepley   Not Collective
309184ce1a2SMatthew G. Knepley 
310184ce1a2SMatthew G. Knepley   Input Parameters:
311184ce1a2SMatthew G. Knepley + handle - obtained with PetscDLOpen() or NULL
312184ce1a2SMatthew G. Knepley - func   - pointer to the function, NULL if not found
313184ce1a2SMatthew G. Knepley 
314184ce1a2SMatthew G. Knepley   Output Parameter:
31504c51a94SMatthew G. Knepley . name   - name of symbol, or NULL if name lookup is not supported.
316184ce1a2SMatthew G. Knepley 
317184ce1a2SMatthew G. Knepley   Level: developer
318184ce1a2SMatthew G. Knepley 
319184ce1a2SMatthew G. Knepley   Notes:
32004c51a94SMatthew G. Knepley   The caller must free the returned name.
321258ec3d2SMatthew G. Knepley 
322184ce1a2SMatthew G. Knepley   In order to be dynamically loadable, the symbol has to be exported as such.  On many UNIX-like
323184ce1a2SMatthew G. Knepley   systems this requires platform-specific linker flags.
32410699b91SBarry Smith 
32510699b91SBarry Smith .seealso: PetscDLClose(), PetscDLSym(), PetscDLOpen()
326184ce1a2SMatthew G. Knepley @*/
327258ec3d2SMatthew G. Knepley PetscErrorCode PetscDLAddr(void (*func)(void), char **name)
328184ce1a2SMatthew G. Knepley {
329184ce1a2SMatthew G. Knepley   PetscFunctionBegin;
330064a246eSJacob Faibussowitsch   PetscValidCharPointer(name,2);
331184ce1a2SMatthew G. Knepley   *name = NULL;
332184ce1a2SMatthew G. Knepley #if defined(PETSC_HAVE_DLADDR)
333184ce1a2SMatthew G. Knepley   dlerror(); /* clear any previous error */
334184ce1a2SMatthew G. Knepley   {
335184ce1a2SMatthew G. Knepley     Dl_info info;
336184ce1a2SMatthew G. Knepley 
3375f80ce2aSJacob Faibussowitsch     PetscCheck(dladdr(*(void **) &func, &info),PETSC_COMM_SELF, PETSC_ERR_LIB, "Failed to lookup symbol: %s", dlerror());
338258ec3d2SMatthew G. Knepley #ifdef PETSC_HAVE_CXX
3395f80ce2aSJacob Faibussowitsch     CHKERRQ(PetscDemangleSymbol(info.dli_sname, name));
340258ec3d2SMatthew G. Knepley #else
3415f80ce2aSJacob Faibussowitsch     CHKERRQ(PetscStrallocpy(info.dli_sname, name));
342258ec3d2SMatthew G. Knepley #endif
343184ce1a2SMatthew G. Knepley   }
344184ce1a2SMatthew G. Knepley #endif
345184ce1a2SMatthew G. Knepley   PetscFunctionReturn(0);
346184ce1a2SMatthew G. Knepley }
347