|  | 
 | /* Support for dynamic loading of extension modules */ | 
 |  | 
 | #include "Python.h" | 
 | #include "importdl.h" | 
 |  | 
 | #include <sys/types.h> | 
 | #include <sys/stat.h> | 
 |  | 
 | #if defined(__NetBSD__) | 
 | #include <sys/param.h> | 
 | #if (NetBSD < 199712) | 
 | #include <nlist.h> | 
 | #include <link.h> | 
 | #define dlerror() "error in dynamic linking" | 
 | #endif | 
 | #endif /* NetBSD */ | 
 |  | 
 | #ifdef HAVE_DLFCN_H | 
 | #include <dlfcn.h> | 
 | #else | 
 | #if defined(PYOS_OS2) && defined(PYCC_GCC) | 
 | #include "dlfcn.h" | 
 | #endif | 
 | #endif | 
 |  | 
 | #if (defined(__OpenBSD__) || defined(__NetBSD__)) && !defined(__ELF__) | 
 | #define LEAD_UNDERSCORE "_" | 
 | #else | 
 | #define LEAD_UNDERSCORE "" | 
 | #endif | 
 |  | 
 |  | 
 | const struct filedescr _PyImport_DynLoadFiletab[] = { | 
 | #ifdef __CYGWIN__ | 
 |     {".dll", "rb", C_EXTENSION}, | 
 |     {"module.dll", "rb", C_EXTENSION}, | 
 | #else | 
 | #if defined(PYOS_OS2) && defined(PYCC_GCC) | 
 |     {".pyd", "rb", C_EXTENSION}, | 
 |     {".dll", "rb", C_EXTENSION}, | 
 | #else | 
 | #ifdef __VMS | 
 |     {".exe", "rb", C_EXTENSION}, | 
 |     {".EXE", "rb", C_EXTENSION}, | 
 |     {"module.exe", "rb", C_EXTENSION}, | 
 |     {"MODULE.EXE", "rb", C_EXTENSION}, | 
 | #else | 
 |     {".so", "rb", C_EXTENSION}, | 
 |     {"module.so", "rb", C_EXTENSION}, | 
 | #endif | 
 | #endif | 
 | #endif | 
 |     {0, 0} | 
 | }; | 
 |  | 
 | static struct { | 
 |     dev_t dev; | 
 | #ifdef __VMS | 
 |     ino_t ino[3]; | 
 | #else | 
 |     ino_t ino; | 
 | #endif | 
 |     void *handle; | 
 | } handles[128]; | 
 | static int nhandles = 0; | 
 |  | 
 |  | 
 | dl_funcptr _PyImport_GetDynLoadFunc(const char *fqname, const char *shortname, | 
 |                                     const char *pathname, FILE *fp) | 
 | { | 
 |     dl_funcptr p; | 
 |     void *handle; | 
 |     char funcname[258]; | 
 |     char pathbuf[260]; | 
 |     int dlopenflags=0; | 
 |  | 
 |     if (strchr(pathname, '/') == NULL) { | 
 |         /* Prefix bare filename with "./" */ | 
 |         PyOS_snprintf(pathbuf, sizeof(pathbuf), "./%-.255s", pathname); | 
 |         pathname = pathbuf; | 
 |     } | 
 |  | 
 |     PyOS_snprintf(funcname, sizeof(funcname), | 
 |                   LEAD_UNDERSCORE "PyInit_%.200s", shortname); | 
 |  | 
 |     if (fp != NULL) { | 
 |         int i; | 
 |         struct stat statb; | 
 |         fstat(fileno(fp), &statb); | 
 |         for (i = 0; i < nhandles; i++) { | 
 |             if (statb.st_dev == handles[i].dev && | 
 |                 statb.st_ino == handles[i].ino) { | 
 |                 p = (dl_funcptr) dlsym(handles[i].handle, | 
 |                                        funcname); | 
 |                 return p; | 
 |             } | 
 |         } | 
 |         if (nhandles < 128) { | 
 |             handles[nhandles].dev = statb.st_dev; | 
 | #ifdef __VMS | 
 |             handles[nhandles].ino[0] = statb.st_ino[0]; | 
 |             handles[nhandles].ino[1] = statb.st_ino[1]; | 
 |             handles[nhandles].ino[2] = statb.st_ino[2]; | 
 | #else | 
 |             handles[nhandles].ino = statb.st_ino; | 
 | #endif | 
 |         } | 
 |     } | 
 |  | 
 | #if !(defined(PYOS_OS2) && defined(PYCC_GCC)) | 
 |     dlopenflags = PyThreadState_GET()->interp->dlopenflags; | 
 | #endif | 
 |  | 
 |     if (Py_VerboseFlag) | 
 |         PySys_WriteStderr("dlopen(\"%s\", %x);\n", pathname, | 
 |                           dlopenflags); | 
 |  | 
 | #ifdef __VMS | 
 |     /* VMS currently don't allow a pathname, use a logical name instead */ | 
 |     /* Concatenate 'python_module_' and shortname */ | 
 |     /* so "import vms.bar" will use the logical python_module_bar */ | 
 |     /* As C module use only one name space this is probably not a */ | 
 |     /* important limitation */ | 
 |     PyOS_snprintf(pathbuf, sizeof(pathbuf), "python_module_%-.200s", | 
 |                   shortname); | 
 |     pathname = pathbuf; | 
 | #endif | 
 |  | 
 |     handle = dlopen(pathname, dlopenflags); | 
 |  | 
 |     if (handle == NULL) { | 
 |         const char *error = dlerror(); | 
 |         if (error == NULL) | 
 |             error = "unknown dlopen() error"; | 
 |         PyErr_SetString(PyExc_ImportError, error); | 
 |         return NULL; | 
 |     } | 
 |     if (fp != NULL && nhandles < 128) | 
 |         handles[nhandles++].handle = handle; | 
 |     p = (dl_funcptr) dlsym(handle, funcname); | 
 |     return p; | 
 | } |