|  | 
 | /* 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 */ |