1999-12-20 21:18:49 +00:00
|
|
|
|
|
|
|
/* Support for dynamic loading of extension modules */
|
|
|
|
|
|
|
|
#include "Python.h"
|
|
|
|
#include "importdl.h"
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
2001-10-18 21:24:04 +00:00
|
|
|
|
2001-10-18 11:45:19 +00:00
|
|
|
#if defined(__NetBSD__)
|
|
|
|
#include <sys/param.h>
|
|
|
|
#if (NetBSD < 199712)
|
1999-12-20 21:18:49 +00:00
|
|
|
#include <nlist.h>
|
|
|
|
#include <link.h>
|
|
|
|
#define dlerror() "error in dynamic linking"
|
2001-10-18 21:24:04 +00:00
|
|
|
#endif
|
|
|
|
#endif /* NetBSD */
|
|
|
|
|
1999-12-20 21:18:49 +00:00
|
|
|
#ifdef HAVE_DLFCN_H
|
|
|
|
#include <dlfcn.h>
|
2002-02-26 11:41:34 +00:00
|
|
|
#else
|
|
|
|
#if defined(PYOS_OS2) && defined(PYCC_GCC)
|
|
|
|
#include "dlfcn.h"
|
|
|
|
#endif
|
1999-12-20 21:18:49 +00:00
|
|
|
#endif
|
|
|
|
|
2001-10-18 11:45:19 +00:00
|
|
|
#if (defined(__OpenBSD__) || defined(__NetBSD__)) && !defined(__ELF__)
|
2000-10-25 22:07:45 +00:00
|
|
|
#define LEAD_UNDERSCORE "_"
|
|
|
|
#else
|
|
|
|
#define LEAD_UNDERSCORE ""
|
|
|
|
#endif
|
|
|
|
|
2010-09-03 18:30:30 +00:00
|
|
|
/* The .so extension module ABI tag, supplied by the Makefile via
|
|
|
|
Makefile.pre.in and configure. This is used to discriminate between
|
|
|
|
incompatible .so files so that extensions for different Python builds can
|
|
|
|
live in the same directory. E.g. foomodule.cpython-32.so
|
|
|
|
*/
|
1999-12-20 21:18:49 +00:00
|
|
|
|
|
|
|
const struct filedescr _PyImport_DynLoadFiletab[] = {
|
2000-10-05 19:24:26 +00:00
|
|
|
#ifdef __CYGWIN__
|
2010-05-09 15:52:27 +00:00
|
|
|
{".dll", "rb", C_EXTENSION},
|
2010-09-03 18:30:30 +00:00
|
|
|
#else /* !__CYGWIN__ */
|
2002-02-26 11:41:34 +00:00
|
|
|
#if defined(PYOS_OS2) && defined(PYCC_GCC)
|
2010-05-09 15:52:27 +00:00
|
|
|
{".pyd", "rb", C_EXTENSION},
|
|
|
|
{".dll", "rb", C_EXTENSION},
|
2010-09-03 18:30:30 +00:00
|
|
|
#else /* !(defined(PYOS_OS2) && defined(PYCC_GCC)) */
|
2003-05-03 09:14:54 +00:00
|
|
|
#ifdef __VMS
|
2010-05-09 15:52:27 +00:00
|
|
|
{".exe", "rb", C_EXTENSION},
|
|
|
|
{".EXE", "rb", C_EXTENSION},
|
2010-09-03 18:30:30 +00:00
|
|
|
#else /* !__VMS */
|
|
|
|
{"." SOABI ".so", "rb", C_EXTENSION},
|
2010-12-03 20:14:31 +00:00
|
|
|
{".abi" PYTHON_ABI_STRING ".so", "rb", C_EXTENSION},
|
2010-09-08 16:22:10 +00:00
|
|
|
{".so", "rb", C_EXTENSION},
|
2010-09-03 18:30:30 +00:00
|
|
|
#endif /* __VMS */
|
|
|
|
#endif /* defined(PYOS_OS2) && defined(PYCC_GCC) */
|
|
|
|
#endif /* __CYGWIN__ */
|
2010-05-09 15:52:27 +00:00
|
|
|
{0, 0}
|
1999-12-20 21:18:49 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct {
|
2010-05-09 15:52:27 +00:00
|
|
|
dev_t dev;
|
2003-05-03 09:14:54 +00:00
|
|
|
#ifdef __VMS
|
2010-05-09 15:52:27 +00:00
|
|
|
ino_t ino[3];
|
2003-05-03 09:14:54 +00:00
|
|
|
#else
|
2010-05-09 15:52:27 +00:00
|
|
|
ino_t ino;
|
2003-05-03 09:14:54 +00:00
|
|
|
#endif
|
2010-05-09 15:52:27 +00:00
|
|
|
void *handle;
|
1999-12-20 21:18:49 +00:00
|
|
|
} handles[128];
|
|
|
|
static int nhandles = 0;
|
|
|
|
|
|
|
|
|
2011-02-22 23:16:19 +00:00
|
|
|
dl_funcptr _PyImport_GetDynLoadFunc(const char *shortname,
|
2010-05-09 15:52:27 +00:00
|
|
|
const char *pathname, FILE *fp)
|
1999-12-20 21:18:49 +00:00
|
|
|
{
|
2010-05-09 15:52:27 +00:00
|
|
|
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;
|
2003-05-03 09:14:54 +00:00
|
|
|
#ifdef __VMS
|
2010-05-09 15:52:27 +00:00
|
|
|
handles[nhandles].ino[0] = statb.st_ino[0];
|
|
|
|
handles[nhandles].ino[1] = statb.st_ino[1];
|
|
|
|
handles[nhandles].ino[2] = statb.st_ino[2];
|
2003-05-03 09:14:54 +00:00
|
|
|
#else
|
2010-05-09 15:52:27 +00:00
|
|
|
handles[nhandles].ino = statb.st_ino;
|
2003-05-03 09:14:54 +00:00
|
|
|
#endif
|
2010-05-09 15:52:27 +00:00
|
|
|
}
|
|
|
|
}
|
1999-12-20 21:18:49 +00:00
|
|
|
|
2002-02-26 11:41:34 +00:00
|
|
|
#if !(defined(PYOS_OS2) && defined(PYCC_GCC))
|
2010-05-09 15:52:27 +00:00
|
|
|
dlopenflags = PyThreadState_GET()->interp->dlopenflags;
|
2002-02-26 11:41:34 +00:00
|
|
|
#endif
|
2001-07-18 16:17:16 +00:00
|
|
|
|
2003-05-03 09:14:54 +00:00
|
|
|
#ifdef __VMS
|
2010-05-09 15:52:27 +00:00
|
|
|
/* 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;
|
2003-05-03 09:14:54 +00:00
|
|
|
#endif
|
|
|
|
|
2010-05-09 15:52:27 +00:00
|
|
|
handle = dlopen(pathname, dlopenflags);
|
|
|
|
|
|
|
|
if (handle == NULL) {
|
2012-04-20 19:22:50 +00:00
|
|
|
PyObject *mod_name = NULL;
|
|
|
|
PyObject *path = NULL;
|
|
|
|
PyObject *error_ob = NULL;
|
2010-05-09 15:52:27 +00:00
|
|
|
const char *error = dlerror();
|
|
|
|
if (error == NULL)
|
|
|
|
error = "unknown dlopen() error";
|
2012-04-20 19:22:50 +00:00
|
|
|
error_ob = PyUnicode_FromString(error);
|
|
|
|
path = PyUnicode_FromString(pathname);
|
|
|
|
mod_name = PyUnicode_FromString(shortname);
|
|
|
|
PyErr_SetImportError(error_ob, mod_name, path);
|
|
|
|
Py_DECREF(error_ob);
|
|
|
|
Py_DECREF(path);
|
|
|
|
Py_DECREF(mod_name);
|
2010-05-09 15:52:27 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (fp != NULL && nhandles < 128)
|
|
|
|
handles[nhandles++].handle = handle;
|
|
|
|
p = (dl_funcptr) dlsym(handle, funcname);
|
|
|
|
return p;
|
1999-12-20 21:18:49 +00:00
|
|
|
}
|