blob: 69bb7117ad82ecf4c59bde663eef3bd4356cdc77 [file] [log] [blame]
Guido van Rossum1ae940a1995-01-02 19:04:15 +00001
2/* Support for dynamic loading of extension modules */
Guido van Rossum1ae940a1995-01-02 19:04:15 +00003
Guido van Rossum79f25d91997-04-29 20:08:16 +00004#include "Python.h"
Guido van Rossum1ae940a1995-01-02 19:04:15 +00005
Guido van Rossum96a8fb71999-12-22 14:09:35 +00006/* ./configure sets HAVE_DYNAMIC_LOADING if dynamic loading of modules is
7 supported on this platform. configure will then compile and link in one
8 of the dynload_*.c files, as appropriate. We will call a function in
9 those modules to get a function pointer to the module's init function.
Guido van Rossum1ae940a1995-01-02 19:04:15 +000010*/
Guido van Rossum6ea90921999-12-20 21:20:42 +000011#ifdef HAVE_DYNAMIC_LOADING
Guido van Rossumff4af061996-01-12 01:17:50 +000012
Guido van Rossum96a8fb71999-12-22 14:09:35 +000013#include "importdl.h"
14
Victor Stinner2d322272011-04-04 23:05:53 +020015#ifdef MS_WINDOWS
16extern dl_funcptr _PyImport_GetDynLoadWindows(const char *shortname,
17 PyObject *pathname, FILE *fp);
18#else
Victor Stinner42040fb2011-02-22 23:16:19 +000019extern dl_funcptr _PyImport_GetDynLoadFunc(const char *shortname,
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000020 const char *pathname, FILE *fp);
Victor Stinner2d322272011-04-04 23:05:53 +020021#endif
Guido van Rossum1ae940a1995-01-02 19:04:15 +000022
Victor Stinnerfefd70c2011-03-14 15:54:07 -040023/* name should be ASCII only because the C language doesn't accept non-ASCII
24 identifiers, and dynamic modules are written in C. */
Guido van Rossum6ea90921999-12-20 21:20:42 +000025
Guido van Rossum79f25d91997-04-29 20:08:16 +000026PyObject *
Victor Stinnerfefd70c2011-03-14 15:54:07 -040027_PyImport_LoadDynamicModule(PyObject *name, PyObject *path, FILE *fp)
Guido van Rossum1ae940a1995-01-02 19:04:15 +000028{
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000029 PyObject *m;
Victor Stinner2d322272011-04-04 23:05:53 +020030#ifndef MS_WINDOWS
Victor Stinnerfefd70c2011-03-14 15:54:07 -040031 PyObject *pathbytes;
Victor Stinner2d322272011-04-04 23:05:53 +020032#endif
Victor Stinnerfefd70c2011-03-14 15:54:07 -040033 char *namestr, *lastdot, *shortname, *packagecontext, *oldcontext;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000034 dl_funcptr p0;
35 PyObject* (*p)(void);
36 struct PyModuleDef *def;
Guido van Rossum6ea90921999-12-20 21:20:42 +000037
Victor Stinnerfefd70c2011-03-14 15:54:07 -040038 namestr = _PyUnicode_AsString(name);
39 if (namestr == NULL)
Victor Stinner49d3f252010-10-17 01:24:53 +000040 return NULL;
41
Victor Stinnerfefd70c2011-03-14 15:54:07 -040042 m = _PyImport_FindExtensionObject(name, path);
Victor Stinner95872862011-03-07 18:20:56 +010043 if (m != NULL) {
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000044 Py_INCREF(m);
Victor Stinnerfefd70c2011-03-14 15:54:07 -040045 return m;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000046 }
Victor Stinnerfefd70c2011-03-14 15:54:07 -040047
48 lastdot = strrchr(namestr, '.');
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000049 if (lastdot == NULL) {
50 packagecontext = NULL;
Victor Stinnerfefd70c2011-03-14 15:54:07 -040051 shortname = namestr;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000052 }
53 else {
Victor Stinnerfefd70c2011-03-14 15:54:07 -040054 packagecontext = namestr;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000055 shortname = lastdot+1;
56 }
Guido van Rossumd5962ad1996-07-31 22:44:53 +000057
Victor Stinner2d322272011-04-04 23:05:53 +020058#ifdef MS_WINDOWS
59 p0 = _PyImport_GetDynLoadWindows(shortname, path, fp);
60#else
Victor Stinnerfefd70c2011-03-14 15:54:07 -040061 pathbytes = PyUnicode_EncodeFSDefault(path);
62 if (pathbytes == NULL)
63 return NULL;
64 p0 = _PyImport_GetDynLoadFunc(shortname,
65 PyBytes_AS_STRING(pathbytes), fp);
66 Py_DECREF(pathbytes);
Victor Stinner2d322272011-04-04 23:05:53 +020067#endif
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000068 p = (PyObject*(*)(void))p0;
69 if (PyErr_Occurred())
Victor Stinnerfefd70c2011-03-14 15:54:07 -040070 return NULL;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000071 if (p == NULL) {
72 PyErr_Format(PyExc_ImportError,
Victor Stinnerfefd70c2011-03-14 15:54:07 -040073 "dynamic module does not define init function"
74 " (PyInit_%s)",
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000075 shortname);
Victor Stinnerfefd70c2011-03-14 15:54:07 -040076 return NULL;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000077 }
78 oldcontext = _Py_PackageContext;
79 _Py_PackageContext = packagecontext;
80 m = (*p)();
81 _Py_PackageContext = oldcontext;
82 if (m == NULL)
Victor Stinnerfefd70c2011-03-14 15:54:07 -040083 return NULL;
Guido van Rossum1ae940a1995-01-02 19:04:15 +000084
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000085 if (PyErr_Occurred()) {
86 Py_DECREF(m);
87 PyErr_Format(PyExc_SystemError,
88 "initialization of %s raised unreported exception",
89 shortname);
Victor Stinnerfefd70c2011-03-14 15:54:07 -040090 return NULL;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000091 }
Martin v. Löwis1a214512008-06-11 05:26:20 +000092
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000093 /* Remember pointer to module init function. */
94 def = PyModule_GetDef(m);
95 def->m_base.m_init = p;
Martin v. Löwis1a214512008-06-11 05:26:20 +000096
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000097 /* Remember the filename as the __file__ attribute */
Antoine Pitrouf95a1b32010-05-09 15:52:27 +000098 if (PyModule_AddObject(m, "__file__", path) < 0)
99 PyErr_Clear(); /* Not important enough to report */
Victor Stinner49d3f252010-10-17 01:24:53 +0000100 else
101 Py_INCREF(path);
Martin v. Löwise81e9b12003-09-04 18:45:59 +0000102
Victor Stinnerfefd70c2011-03-14 15:54:07 -0400103 if (_PyImport_FixupExtensionObject(m, name, path) < 0)
Victor Stinner95872862011-03-07 18:20:56 +0100104 return NULL;
Antoine Pitrouf95a1b32010-05-09 15:52:27 +0000105 if (Py_VerboseFlag)
Victor Stinner95872862011-03-07 18:20:56 +0100106 PySys_FormatStderr(
Victor Stinnerfefd70c2011-03-14 15:54:07 -0400107 "import %U # dynamically loaded from %R\n",
108 name, path);
109 return m;
Guido van Rossum1ae940a1995-01-02 19:04:15 +0000110}
Guido van Rossum96a8fb71999-12-22 14:09:35 +0000111
112#endif /* HAVE_DYNAMIC_LOADING */