| |
| /* Support for dynamic loading of extension modules */ |
| |
| #include "Python.h" |
| |
| /* ./configure sets HAVE_DYNAMIC_LOADING if dynamic loading of modules is |
| supported on this platform. configure will then compile and link in one |
| of the dynload_*.c files, as appropriate. We will call a function in |
| those modules to get a function pointer to the module's init function. |
| */ |
| #ifdef HAVE_DYNAMIC_LOADING |
| |
| #include "importdl.h" |
| |
| extern dl_funcptr _PyImport_GetDynLoadFunc(const char *name, |
| const char *shortname, |
| const char *pathname, FILE *fp); |
| |
| |
| |
| PyObject * |
| _PyImport_LoadDynamicModule(char *name, char *pathname, FILE *fp) |
| { |
| PyObject *m; |
| PyObject *path; |
| char *lastdot, *shortname, *packagecontext, *oldcontext; |
| dl_funcptr p0; |
| PyObject* (*p)(void); |
| struct PyModuleDef *def; |
| PyObject *result; |
| |
| path = PyUnicode_DecodeFSDefault(pathname); |
| if (path == NULL) |
| return NULL; |
| |
| if ((m = _PyImport_FindExtensionUnicode(name, path)) != NULL) { |
| Py_INCREF(m); |
| result = m; |
| goto finally; |
| } |
| lastdot = strrchr(name, '.'); |
| if (lastdot == NULL) { |
| packagecontext = NULL; |
| shortname = name; |
| } |
| else { |
| packagecontext = name; |
| shortname = lastdot+1; |
| } |
| |
| p0 = _PyImport_GetDynLoadFunc(name, shortname, pathname, fp); |
| p = (PyObject*(*)(void))p0; |
| if (PyErr_Occurred()) |
| goto error; |
| if (p == NULL) { |
| PyErr_Format(PyExc_ImportError, |
| "dynamic module does not define init function (PyInit_%.200s)", |
| shortname); |
| goto error; |
| } |
| oldcontext = _Py_PackageContext; |
| _Py_PackageContext = packagecontext; |
| m = (*p)(); |
| _Py_PackageContext = oldcontext; |
| if (m == NULL) |
| goto error; |
| |
| if (PyErr_Occurred()) { |
| Py_DECREF(m); |
| PyErr_Format(PyExc_SystemError, |
| "initialization of %s raised unreported exception", |
| shortname); |
| goto error; |
| } |
| |
| /* Remember pointer to module init function. */ |
| def = PyModule_GetDef(m); |
| def->m_base.m_init = p; |
| |
| /* Remember the filename as the __file__ attribute */ |
| if (PyModule_AddObject(m, "__file__", path) < 0) |
| PyErr_Clear(); /* Not important enough to report */ |
| else |
| Py_INCREF(path); |
| |
| if (_PyImport_FixupExtensionUnicode(m, name, path) < 0) |
| goto error; |
| if (Py_VerboseFlag) |
| PySys_WriteStderr( |
| "import %s # dynamically loaded from %s\n", |
| name, pathname); |
| result = m; |
| goto finally; |
| |
| error: |
| result = NULL; |
| finally: |
| Py_DECREF(path); |
| return result; |
| } |
| |
| #endif /* HAVE_DYNAMIC_LOADING */ |