17d0a6c19SBarry Smith 2ebd79076SLisandro Dalcin /* 3ebd79076SLisandro Dalcin Low-level routines for managing dynamic link libraries (DLLs). 4ebd79076SLisandro Dalcin */ 5ebd79076SLisandro Dalcin 6c6db04a5SJed Brown #include <../src/sys/dll/dlimpl.h> 7ebd79076SLisandro Dalcin 8ebd79076SLisandro Dalcin /* XXX Should be done better !!!*/ 9be1c6ad7SLisandro Dalcin #if !defined(PETSC_HAVE_DYNAMIC_LIBRARIES) 10ebd79076SLisandro Dalcin #undef PETSC_HAVE_WINDOWS_H 11ebd79076SLisandro Dalcin #undef PETSC_HAVE_DLFCN_H 12ebd79076SLisandro Dalcin #endif 13ebd79076SLisandro Dalcin 14ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_WINDOWS_H) 15ebd79076SLisandro Dalcin #include <windows.h> 16be1c6ad7SLisandro Dalcin #elif defined(PETSC_HAVE_DLFCN_H) 17ebd79076SLisandro Dalcin #include <dlfcn.h> 18ebd79076SLisandro Dalcin #endif 19ebd79076SLisandro Dalcin 20ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_WINDOWS_H) 21ebd79076SLisandro Dalcin typedef HMODULE dlhandle_t; 226ea75d68SLisandro Dalcin typedef FARPROC dlsymbol_t; 23ebd79076SLisandro Dalcin #elif defined(PETSC_HAVE_DLFCN_H) 24ebd79076SLisandro Dalcin typedef void* dlhandle_t; 256ea75d68SLisandro Dalcin typedef void* dlsymbol_t; 26ebd79076SLisandro Dalcin #else 27ebd79076SLisandro Dalcin typedef void* dlhandle_t; 286ea75d68SLisandro Dalcin typedef void* dlsymbol_t; 29ebd79076SLisandro Dalcin #endif 30ebd79076SLisandro Dalcin 31ebd79076SLisandro Dalcin #undef __FUNCT__ 32ebd79076SLisandro Dalcin #define __FUNCT__ "PetscDLOpen" 33ebd79076SLisandro Dalcin /*@C 34b235ab32SBarry Smith PetscDLOpen - opens dynamic library 35b235ab32SBarry Smith 36b235ab32SBarry Smith Not Collective 37b235ab32SBarry Smith 38b235ab32SBarry Smith Input Parameters: 39b235ab32SBarry Smith + name - name of library 40b235ab32SBarry Smith - flags - options on how to open library 41b235ab32SBarry Smith 42b235ab32SBarry Smith Output Parameter: 43b235ab32SBarry Smith . handle 44b235ab32SBarry Smith 45b235ab32SBarry Smith Level: developer 46b235ab32SBarry Smith 47ebd79076SLisandro Dalcin @*/ 487087cfbeSBarry Smith PetscErrorCode PetscDLOpen(const char name[],int flags,PetscDLHandle *handle) 49ebd79076SLisandro Dalcin { 50be1c6ad7SLisandro Dalcin int dlflags1,dlflags2; 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, 74ebd79076SLisandro Dalcin NULL,erc,MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),(LPSTR)&buff,0,NULL); 7530b327c4SMatthew G Knepley ierr = PetscError(PETSC_COMM_SELF,__LINE__,__FUNCT__,__FILE__,__SDIR__,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 80e32f2f54SBarry Smith SETERRQ2(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to open dynamic library:\n %s\n Error message from LoadLibrary() %s\n",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) 985673baf8SLisandro Dalcin if (flags & PETSC_DL_NOW) 99be1c6ad7SLisandro Dalcin dlflags1 = RTLD_NOW; 100ebd79076SLisandro Dalcin #endif 101ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_RTLD_GLOBAL) 102be1c6ad7SLisandro Dalcin dlflags2 = RTLD_GLOBAL; 103ebd79076SLisandro Dalcin #endif 104b3bb0f5eSLisandro Dalcin #if defined(PETSC_HAVE_RTLD_LOCAL) 105b3bb0f5eSLisandro Dalcin if (flags & PETSC_DL_LOCAL) 106b3bb0f5eSLisandro Dalcin dlflags2 = RTLD_LOCAL; 107b3bb0f5eSLisandro Dalcin #endif 108be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR) 109be1c6ad7SLisandro Dalcin dlerror(); /* clear any previous error */ 110be1c6ad7SLisandro Dalcin #endif 111be1c6ad7SLisandro Dalcin dlhandle = dlopen(name,dlflags1|dlflags2); 112ebd79076SLisandro Dalcin if (!dlhandle) { 113ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR) 114ebd79076SLisandro Dalcin const char *errmsg = dlerror(); 115ebd79076SLisandro Dalcin #else 116ebd79076SLisandro Dalcin const char *errmsg = "unavailable"; 117ebd79076SLisandro Dalcin #endif 118be015529SBarry Smith SETERRQ2(PETSC_COMM_SELF,PETSC_ERR_FILE_OPEN,"Unable to open dynamic library:\n %s\n Error message from dlopen() %s\n",name,errmsg); 119ebd79076SLisandro Dalcin } 1205673baf8SLisandro Dalcin 1215673baf8SLisandro Dalcin /* 1225673baf8SLisandro Dalcin --- unimplemented --- 1235673baf8SLisandro Dalcin */ 124ebd79076SLisandro Dalcin #else 125e32f2f54SBarry Smith SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS, "Cannot use dynamic libraries on this platform"); 126ebd79076SLisandro Dalcin #endif 127ebd79076SLisandro Dalcin 128ebd79076SLisandro Dalcin *handle = (PetscDLHandle) dlhandle; 129ebd79076SLisandro Dalcin 130ebd79076SLisandro Dalcin PetscFunctionReturn(0); 131ebd79076SLisandro Dalcin } 132ebd79076SLisandro Dalcin 133ebd79076SLisandro Dalcin 134ebd79076SLisandro Dalcin #undef __FUNCT__ 135ebd79076SLisandro Dalcin #define __FUNCT__ "PetscDLClose" 136ebd79076SLisandro Dalcin /*@C 137b235ab32SBarry Smith PetscDLClose - closes a dynamic library 138b235ab32SBarry Smith 139b235ab32SBarry Smith Not Collective 140b235ab32SBarry Smith 141b235ab32SBarry Smith Input Parameter: 142b235ab32SBarry Smith . handle - the handle for the library obtained with PetscDLOpen() 143b235ab32SBarry Smith 144b235ab32SBarry Smith Level: developer 145ebd79076SLisandro Dalcin @*/ 1467087cfbeSBarry Smith PetscErrorCode PetscDLClose(PetscDLHandle *handle) 147ebd79076SLisandro Dalcin { 148ebd79076SLisandro Dalcin dlhandle_t dlhandle; 149ebd79076SLisandro Dalcin 150ebd79076SLisandro Dalcin PetscFunctionBegin; 151ebd79076SLisandro Dalcin PetscValidPointer(handle,1); 152ebd79076SLisandro Dalcin 153ebd79076SLisandro Dalcin dlhandle = (dlhandle_t) *handle; 154ebd79076SLisandro Dalcin 1555673baf8SLisandro Dalcin /* 1565673baf8SLisandro Dalcin --- FreeLibrary --- 1575673baf8SLisandro Dalcin */ 158ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_WINDOWS_H) 159be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_FREELIBRARY) 1605673baf8SLisandro Dalcin if (FreeLibrary(dlhandle) == 0) { 1615673baf8SLisandro Dalcin #if defined(PETSC_HAVE_GETLASTERROR) 162b3bb0f5eSLisandro Dalcin char *buff = NULL; 163a21658a3SLisandro Dalcin DWORD erc = GetLastError(); 1645673baf8SLisandro Dalcin FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_IGNORE_INSERTS, 1655673baf8SLisandro Dalcin NULL,erc,MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),(LPSTR)&buff,0,NULL); 1665673baf8SLisandro Dalcin PetscErrorPrintf("Error closing dynamic library:\n Error message from FreeLibrary() %s\n",buff); 1675673baf8SLisandro Dalcin LocalFree(buff); 1685673baf8SLisandro Dalcin #else 1695673baf8SLisandro Dalcin PetscErrorPrintf("Error closing dynamic library:\n Error message from FreeLibrary() %s\n","unavailable"); 1705673baf8SLisandro Dalcin #endif 1715673baf8SLisandro Dalcin } 172be1c6ad7SLisandro Dalcin #endif /* !PETSC_HAVE_FREELIBRARY */ 173ebd79076SLisandro Dalcin 1745673baf8SLisandro Dalcin /* 1755673baf8SLisandro Dalcin --- dclose --- 1765673baf8SLisandro Dalcin */ 177ebd79076SLisandro Dalcin #elif defined(PETSC_HAVE_DLFCN_H) 178a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_DLCLOSE) 179be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR) 180be1c6ad7SLisandro Dalcin dlerror(); /* clear any previous error */ 181be1c6ad7SLisandro Dalcin #endif 182ebd79076SLisandro Dalcin if (dlclose(dlhandle) < 0) { 183ebd79076SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR) 184ebd79076SLisandro Dalcin const char *errmsg = dlerror(); 185ebd79076SLisandro Dalcin #else 186ebd79076SLisandro Dalcin const char *errmsg = "unavailable"; 187ebd79076SLisandro Dalcin #endif 1885673baf8SLisandro Dalcin PetscErrorPrintf("Error closing dynamic library:\n Error message from dlclose() %s\n", errmsg); 189ebd79076SLisandro Dalcin } 190a21658a3SLisandro Dalcin #endif /* !PETSC_HAVE_DLCLOSE */ 1915673baf8SLisandro Dalcin 1925673baf8SLisandro Dalcin /* 1935673baf8SLisandro Dalcin --- unimplemented --- 1945673baf8SLisandro Dalcin */ 195ebd79076SLisandro Dalcin #else 196e32f2f54SBarry Smith SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS, "Cannot use dynamic libraries on this platform"); 197ebd79076SLisandro Dalcin #endif 198ebd79076SLisandro Dalcin 199ebd79076SLisandro Dalcin *handle = PETSC_NULL; 200ebd79076SLisandro Dalcin 201ebd79076SLisandro Dalcin PetscFunctionReturn(0); 202ebd79076SLisandro Dalcin } 203ebd79076SLisandro Dalcin 2047bb14e67SBarry Smith #if defined(PETSC_HAVE_VALGRIND) 2057bb14e67SBarry Smith #include <valgrind/valgrind.h> 2067bb14e67SBarry Smith #endif 207ebd79076SLisandro Dalcin 208ebd79076SLisandro Dalcin #undef __FUNCT__ 209ebd79076SLisandro Dalcin #define __FUNCT__ "PetscDLSym" 210ebd79076SLisandro Dalcin /*@C 211b235ab32SBarry Smith PetscDLSym - finds a symbol in a dynamic library 212b235ab32SBarry Smith 213b235ab32SBarry Smith Not Collective 214b235ab32SBarry Smith 215b235ab32SBarry Smith Input Parameters: 2167c62f5d3SDmitry Karpeev + handle - obtained with PetscDLOpen() or PETSC_NULL 217b235ab32SBarry Smith - symbol - name of symbol 218b235ab32SBarry Smith 219b235ab32SBarry Smith Output Parameter: 220b235ab32SBarry Smith . value - pointer to the function 221b235ab32SBarry Smith 222b235ab32SBarry Smith Level: developer 223b235ab32SBarry Smith 2247c62f5d3SDmitry Karpeev Notes: 2257c62f5d3SDmitry Karpeev If handle is PETSC_NULL, the symbol is looked for in the main executable's dynamic symbol table. 2267c62f5d3SDmitry Karpeev In order to be dynamically loadable, the symbol has to be exported as such. On many UNIX-like 2277c62f5d3SDmitry Karpeev systems this requires platform-specific linker flags. 2287c62f5d3SDmitry Karpeev 229ebd79076SLisandro Dalcin @*/ 2307087cfbeSBarry Smith PetscErrorCode PetscDLSym(PetscDLHandle handle,const char symbol[],void **value) 231ebd79076SLisandro Dalcin { 232ebd79076SLisandro Dalcin dlhandle_t dlhandle; 2336ea75d68SLisandro Dalcin dlsymbol_t dlsymbol; 234ebd79076SLisandro Dalcin 235ebd79076SLisandro Dalcin PetscFunctionBegin; 236ebd79076SLisandro Dalcin PetscValidCharPointer(symbol,2); 237ebd79076SLisandro Dalcin PetscValidPointer(value,3); 238ebd79076SLisandro Dalcin 239be1c6ad7SLisandro Dalcin dlhandle = (dlhandle_t) 0; 2406ea75d68SLisandro Dalcin dlsymbol = (dlsymbol_t) 0; 2416ea75d68SLisandro Dalcin 242a21658a3SLisandro Dalcin *value = (void *) 0; 243ebd79076SLisandro Dalcin 2445673baf8SLisandro Dalcin /* 2455673baf8SLisandro Dalcin --- GetProcAddress --- 2465673baf8SLisandro Dalcin */ 247a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_WINDOWS_H) 248a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_GETPROCADDRESS) 2497d421530SBarry Smith if (handle) 250be1c6ad7SLisandro Dalcin dlhandle = (dlhandle_t) handle; 251be1c6ad7SLisandro Dalcin else 252be1c6ad7SLisandro Dalcin dlhandle = (dlhandle_t) GetCurrentProcess(); 2536ea75d68SLisandro Dalcin dlsymbol = (dlsymbol_t) GetProcAddress(dlhandle,symbol); 254a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_SETLASTERROR) 255a21658a3SLisandro Dalcin SetLastError((DWORD)0); /* clear any previous error */ 256a21658a3SLisandro Dalcin #endif 257a21658a3SLisandro Dalcin #endif /* !PETSC_HAVE_GETPROCADDRESS */ 2585673baf8SLisandro Dalcin 2595673baf8SLisandro Dalcin /* 2605673baf8SLisandro Dalcin --- dlsym --- 2615673baf8SLisandro Dalcin */ 262ebd79076SLisandro Dalcin #elif defined(PETSC_HAVE_DLFCN_H) 263a21658a3SLisandro Dalcin #if defined(PETSC_HAVE_DLSYM) 2647d421530SBarry Smith if (handle) { 265be1c6ad7SLisandro Dalcin dlhandle = (dlhandle_t) handle; 2667d421530SBarry Smith } else { 2677d421530SBarry Smith 2687c62f5d3SDmitry Karpeev 269*bfef2c86SBarry Smith #if defined(PETSC_HAVE_DLOPEN) && defined(PETSC_HAVE_DYNAMIC_LIBRARIES) 2707c62f5d3SDmitry Karpeev /* Attempt to retrieve the main executable's dlhandle. */ 2717c62f5d3SDmitry Karpeev { int dlflags1 = 0, dlflags2 = 0; 2727c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_RTLD_LAZY) 2737c62f5d3SDmitry Karpeev dlflags1 = RTLD_LAZY; 2747c62f5d3SDmitry Karpeev #endif 2757c62f5d3SDmitry Karpeev if (!dlflags1) { 2767c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_RTLD_NOW) 2777c62f5d3SDmitry Karpeev dlflags1 = RTLD_NOW; 2787c62f5d3SDmitry Karpeev #endif 2797c62f5d3SDmitry Karpeev } 2807c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_RTLD_LOCAL) 2817c62f5d3SDmitry Karpeev dlflags2 = RTLD_LOCAL; 2827c62f5d3SDmitry Karpeev #endif 2837c62f5d3SDmitry Karpeev if(!dlflags2) { 2847c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_RTLD_GLOBAL) 2857c62f5d3SDmitry Karpeev dlflags2 = RTLD_GLOBAL; 2867c62f5d3SDmitry Karpeev #endif 2877c62f5d3SDmitry Karpeev } 2887c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_DLERROR) 2897bb14e67SBarry Smith #if defined(PETSC_HAVE_VALGRIND) 2907bb14e67SBarry Smith if (!(RUNNING_ON_VALGRIND)) { 2917bb14e67SBarry Smith #endif 2927bb14e67SBarry Smith dlerror(); /* clear any previous error; valgrind does not like this */ 2937bb14e67SBarry Smith #if defined(PETSC_HAVE_VALGRIND) 2947bb14e67SBarry Smith } 2957bb14e67SBarry Smith #endif 2967c62f5d3SDmitry Karpeev #endif 2977c62f5d3SDmitry Karpeev /* Attempt to open the main executable as a dynamic library. */ 2987d421530SBarry Smith #if defined(PETSC_HAVE_RTDL_DEFAULT) 2997d421530SBarry Smith dlhandle = RTLD_DEFAULT; 3007d421530SBarry Smith #else 3017c62f5d3SDmitry Karpeev dlhandle = dlopen(0, dlflags1|dlflags2); 3027c62f5d3SDmitry Karpeev #if defined(PETSC_HAVE_DLERROR) 3034956914eSSatish Balay { const char *e = (const char*) dlerror(); 3047d421530SBarry Smith if (e) SETERRQ1(PETSC_COMM_SELF, PETSC_ERR_ARG_WRONG, "Error opening main executable as a dynamic library:\n Error message from dlopen(): '%s'\n", e); 3057c62f5d3SDmitry Karpeev } 3067c62f5d3SDmitry Karpeev #endif 3077d421530SBarry Smith #endif 3087d421530SBarry Smith } 3093d61445fSSatish Balay #endif /* PETSC_HAVE_DLOPEN && PETSC_USE_DYNAMIC_LIBRARIES */ 3104956914eSSatish Balay } 311be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR) 312be1c6ad7SLisandro Dalcin dlerror(); /* clear any previous error */ 313be1c6ad7SLisandro Dalcin #endif 3146ea75d68SLisandro Dalcin dlsymbol = (dlsymbol_t) dlsym(dlhandle,symbol); 315be1c6ad7SLisandro Dalcin #if defined(PETSC_HAVE_DLERROR) 3167d421530SBarry Smith { const char *e = dlerror(); 3177d421530SBarry Smith if (e) SETERRQ2(PETSC_COMM_SELF, PETSC_ERR_ARG_WRONG,"Error obtaining symbol '%s' from dynamic library:\n Error message from dlsym(): '%s'\n", symbol, e); 3187d421530SBarry Smith } 319be1c6ad7SLisandro Dalcin #endif 320a21658a3SLisandro Dalcin #endif /* !PETSC_HAVE_DLSYM */ 3215673baf8SLisandro Dalcin 3225673baf8SLisandro Dalcin /* 3235673baf8SLisandro Dalcin --- unimplemented --- 3245673baf8SLisandro Dalcin */ 325ebd79076SLisandro Dalcin #else 326e32f2f54SBarry Smith SETERRQ(PETSC_COMM_SELF,PETSC_ERR_SUP_SYS, "Cannot use dynamic libraries on this platform"); 327ebd79076SLisandro Dalcin #endif 328ebd79076SLisandro Dalcin 3296ea75d68SLisandro Dalcin *value = *((void**)&dlsymbol); 330ebd79076SLisandro Dalcin 331ebd79076SLisandro Dalcin PetscFunctionReturn(0); 332ebd79076SLisandro Dalcin } 333