| 
									
										
										
										
											1995-01-02 19:04:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Support for dynamic loading of extension modules */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-04-29 20:08:16 +00:00
										 |  |  | #include "Python.h"
 | 
					
						
							| 
									
										
										
										
											2022-02-08 13:39:07 -07:00
										 |  |  | #include "pycore_call.h"
 | 
					
						
							| 
									
										
										
										
											2023-02-15 18:16:00 -07:00
										 |  |  | #include "pycore_import.h"
 | 
					
						
							| 
									
										
										
										
											2023-07-03 12:48:50 +02:00
										 |  |  | #include "pycore_pyerrors.h"      // _PyErr_FormatFromCause()
 | 
					
						
							| 
									
										
										
										
											2022-02-08 13:39:07 -07:00
										 |  |  | #include "pycore_pystate.h"
 | 
					
						
							|  |  |  | #include "pycore_runtime.h"
 | 
					
						
							| 
									
										
										
										
											1995-01-02 19:04:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-22 14:09:35 +00:00
										 |  |  | /* ./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. | 
					
						
							| 
									
										
										
										
											1995-01-02 19:04:15 +00:00
										 |  |  | */ | 
					
						
							| 
									
										
										
										
											1999-12-20 21:20:42 +00:00
										 |  |  | #ifdef HAVE_DYNAMIC_LOADING
 | 
					
						
							| 
									
										
										
										
											1995-01-02 19:04:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-04 22:50:29 +00:00
										 |  |  | #include "pycore_importdl.h"
 | 
					
						
							| 
									
										
										
										
											1999-12-22 14:09:35 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-04 23:05:53 +02:00
										 |  |  | #ifdef MS_WINDOWS
 | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | extern dl_funcptr _PyImport_FindSharedFuncptrWindows(const char *prefix, | 
					
						
							|  |  |  |                                                      const char *shortname, | 
					
						
							|  |  |  |                                                      PyObject *pathname, | 
					
						
							|  |  |  |                                                      FILE *fp); | 
					
						
							| 
									
										
										
										
											2011-04-04 23:05:53 +02:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | extern dl_funcptr _PyImport_FindSharedFuncptr(const char *prefix, | 
					
						
							|  |  |  |                                               const char *shortname, | 
					
						
							|  |  |  |                                               const char *pathname, FILE *fp); | 
					
						
							| 
									
										
										
										
											2011-04-04 23:05:53 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1997-11-22 21:53:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-25 19:53:18 +02:00
										 |  |  | static const char * const ascii_only_prefix = "PyInit"; | 
					
						
							|  |  |  | static const char * const nonascii_prefix = "PyInitU"; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Get the variable part of a module's export symbol name.
 | 
					
						
							|  |  |  |  * Returns a bytes instance. For non-ASCII-named modules, the name is | 
					
						
							|  |  |  |  * encoded as per PEP 489. | 
					
						
							|  |  |  |  * The hook_prefix pointer is set to either ascii_only_prefix or | 
					
						
							|  |  |  |  * nonascii_prefix, as appropriate. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static PyObject * | 
					
						
							|  |  |  | get_encoded_name(PyObject *name, const char **hook_prefix) { | 
					
						
							|  |  |  |     PyObject *tmp; | 
					
						
							|  |  |  |     PyObject *encoded = NULL; | 
					
						
							| 
									
										
										
										
											2015-05-29 17:10:30 -05:00
										 |  |  |     PyObject *modname = NULL; | 
					
						
							|  |  |  |     Py_ssize_t name_len, lastdot; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Get the short name (substring after last dot) */ | 
					
						
							|  |  |  |     name_len = PyUnicode_GetLength(name); | 
					
						
							| 
									
										
										
										
											2021-10-11 20:08:38 +09:00
										 |  |  |     if (name_len < 0) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     lastdot = PyUnicode_FindChar(name, '.', 0, name_len, -1); | 
					
						
							|  |  |  |     if (lastdot < -1) { | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     } else if (lastdot >= 0) { | 
					
						
							| 
									
										
										
										
											2015-05-26 21:48:17 +10:00
										 |  |  |         tmp = PyUnicode_Substring(name, lastdot + 1, name_len); | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |         if (tmp == NULL) | 
					
						
							|  |  |  |             return NULL; | 
					
						
							|  |  |  |         name = tmp; | 
					
						
							|  |  |  |         /* "name" now holds a new reference to the substring */ | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         Py_INCREF(name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Encode to ASCII or Punycode, as needed */ | 
					
						
							|  |  |  |     encoded = PyUnicode_AsEncodedString(name, "ascii", NULL); | 
					
						
							|  |  |  |     if (encoded != NULL) { | 
					
						
							|  |  |  |         *hook_prefix = ascii_only_prefix; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         if (PyErr_ExceptionMatches(PyExc_UnicodeEncodeError)) { | 
					
						
							|  |  |  |             PyErr_Clear(); | 
					
						
							|  |  |  |             encoded = PyUnicode_AsEncodedString(name, "punycode", NULL); | 
					
						
							|  |  |  |             if (encoded == NULL) { | 
					
						
							|  |  |  |                 goto error; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             *hook_prefix = nonascii_prefix; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             goto error; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-29 17:10:30 -05:00
										 |  |  |     /* Replace '-' by '_' */ | 
					
						
							| 
									
										
										
										
											2022-02-08 13:39:07 -07:00
										 |  |  |     modname = _PyObject_CallMethod(encoded, &_Py_ID(replace), "cc", '-', '_'); | 
					
						
							| 
									
										
										
										
											2015-05-29 17:10:30 -05:00
										 |  |  |     if (modname == NULL) | 
					
						
							|  |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Py_DECREF(name); | 
					
						
							| 
									
										
										
										
											2015-05-29 17:10:30 -05:00
										 |  |  |     Py_DECREF(encoded); | 
					
						
							|  |  |  |     return modname; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | error: | 
					
						
							|  |  |  |     Py_DECREF(name); | 
					
						
							|  |  |  |     Py_XDECREF(encoded); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-04-29 20:08:16 +00:00
										 |  |  | PyObject * | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | _PyImport_LoadDynamicModuleWithSpec(PyObject *spec, FILE *fp) | 
					
						
							| 
									
										
										
										
											1995-01-02 19:04:15 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-04-04 23:05:53 +02:00
										 |  |  | #ifndef MS_WINDOWS
 | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |     PyObject *filename_bytes = NULL; | 
					
						
							|  |  |  |     const char *filename_buf; | 
					
						
							| 
									
										
										
										
											2011-04-04 23:05:53 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |     PyObject *name_unicode = NULL, *name = NULL, *filename = NULL, *m = NULL; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     const char *name_buf, *hook_prefix; | 
					
						
							| 
									
										
										
										
											2023-02-15 15:32:31 -07:00
										 |  |  |     const char *oldcontext, *newcontext; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     dl_funcptr exportfunc; | 
					
						
							|  |  |  |     PyModuleDef *def; | 
					
						
							| 
									
										
										
										
											2022-03-30 22:28:33 +03:00
										 |  |  |     PyModInitFunction p0; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     name_unicode = PyObject_GetAttrString(spec, "name"); | 
					
						
							|  |  |  |     if (name_unicode == NULL) { | 
					
						
							| 
									
										
										
										
											2011-05-07 12:46:05 +02:00
										 |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-09-19 14:39:47 +03:00
										 |  |  |     if (!PyUnicode_Check(name_unicode)) { | 
					
						
							|  |  |  |         PyErr_SetString(PyExc_TypeError, | 
					
						
							|  |  |  |                         "spec.name must be a string"); | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-15 15:32:31 -07:00
										 |  |  |     newcontext = PyUnicode_AsUTF8(name_unicode); | 
					
						
							|  |  |  |     if (newcontext == NULL) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-05-07 12:46:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     name = get_encoded_name(name_unicode, &hook_prefix); | 
					
						
							|  |  |  |     if (name == NULL) { | 
					
						
							| 
									
										
										
										
											2011-05-07 12:46:05 +02:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     name_buf = PyBytes_AS_STRING(name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |     filename = PyObject_GetAttrString(spec, "origin"); | 
					
						
							|  |  |  |     if (filename == NULL) { | 
					
						
							| 
									
										
										
										
											2021-10-11 11:57:27 +03:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2019-05-23 08:45:22 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-04 23:05:53 +02:00
										 |  |  | #ifdef MS_WINDOWS
 | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |     exportfunc = _PyImport_FindSharedFuncptrWindows( | 
					
						
							|  |  |  |             hook_prefix, name_buf, filename, fp); | 
					
						
							| 
									
										
										
										
											2011-04-04 23:05:53 +02:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |     filename_bytes = PyUnicode_EncodeFSDefault(filename); | 
					
						
							|  |  |  |     if (filename_bytes == NULL) { | 
					
						
							| 
									
										
										
										
											2011-05-07 12:46:05 +02:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |     } | 
					
						
							|  |  |  |     filename_buf = PyBytes_AS_STRING(filename_bytes); | 
					
						
							|  |  |  |     exportfunc = _PyImport_FindSharedFuncptr( | 
					
						
							|  |  |  |             hook_prefix, name_buf, filename_buf, fp); | 
					
						
							|  |  |  |     Py_DECREF(filename_bytes); | 
					
						
							| 
									
										
										
										
											2011-04-04 23:05:53 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (exportfunc == NULL) { | 
					
						
							|  |  |  |         if (!PyErr_Occurred()) { | 
					
						
							|  |  |  |             PyObject *msg; | 
					
						
							|  |  |  |             msg = PyUnicode_FromFormat( | 
					
						
							|  |  |  |                 "dynamic module does not define " | 
					
						
							|  |  |  |                 "module export function (%s_%s)", | 
					
						
							|  |  |  |                 hook_prefix, name_buf); | 
					
						
							|  |  |  |             if (msg == NULL) | 
					
						
							|  |  |  |                 goto error; | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |             PyErr_SetImportError(msg, name_unicode, filename); | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |             Py_DECREF(msg); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-05-07 12:46:05 +02:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-30 22:28:33 +03:00
										 |  |  |     p0 = (PyModInitFunction)exportfunc; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* Package context is needed for single-phase init */ | 
					
						
							| 
									
										
										
										
											2023-02-15 15:32:31 -07:00
										 |  |  |     oldcontext = _PyImport_SwapPackageContext(newcontext); | 
					
						
							| 
									
										
										
										
											2023-06-29 17:20:49 -07:00
										 |  |  |     m = p0(); | 
					
						
							| 
									
										
										
										
											2023-02-15 15:32:31 -07:00
										 |  |  |     _PyImport_SwapPackageContext(oldcontext); | 
					
						
							| 
									
										
										
										
											1995-01-02 19:04:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     if (m == NULL) { | 
					
						
							|  |  |  |         if (!PyErr_Occurred()) { | 
					
						
							|  |  |  |             PyErr_Format( | 
					
						
							|  |  |  |                 PyExc_SystemError, | 
					
						
							|  |  |  |                 "initialization of %s failed without raising an exception", | 
					
						
							|  |  |  |                 name_buf); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } else if (PyErr_Occurred()) { | 
					
						
							| 
									
										
										
										
											2022-12-24 00:43:19 +01:00
										 |  |  |         _PyErr_FormatFromCause( | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |             PyExc_SystemError, | 
					
						
							|  |  |  |             "initialization of %s raised unreported exception", | 
					
						
							|  |  |  |             name_buf); | 
					
						
							|  |  |  |         m = NULL; | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-04 07:15:20 -06:00
										 |  |  |     if (Py_IS_TYPE(m, NULL)) { | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |         /* This can happen when a PyModuleDef is returned without calling
 | 
					
						
							|  |  |  |          * PyModuleDef_Init on it | 
					
						
							|  |  |  |          */ | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |         PyErr_Format(PyExc_SystemError, | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |                      "init function of %s returned uninitialized object", | 
					
						
							|  |  |  |                      name_buf); | 
					
						
							|  |  |  |         m = NULL; /* prevent segfault in DECREF */ | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (PyObject_TypeCheck(m, &PyModuleDef_Type)) { | 
					
						
							|  |  |  |         Py_DECREF(name_unicode); | 
					
						
							|  |  |  |         Py_DECREF(name); | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |         Py_DECREF(filename); | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |         return PyModule_FromDefAndSpec((PyModuleDef*)m, spec); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Fall back to single-phase init mechanism */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-15 18:16:00 -07:00
										 |  |  |     if (_PyImport_CheckSubinterpIncompatibleExtensionAllowed(name_buf) < 0) { | 
					
						
							|  |  |  |         goto error; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     if (hook_prefix == nonascii_prefix) { | 
					
						
							|  |  |  |         /* don't allow legacy init for non-ASCII module names */ | 
					
						
							|  |  |  |         PyErr_Format( | 
					
						
							|  |  |  |             PyExc_SystemError, | 
					
						
							| 
									
										
										
										
											2021-10-12 10:20:04 +03:00
										 |  |  |             "initialization of %s did not return PyModuleDef", | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |             name_buf); | 
					
						
							| 
									
										
										
										
											2011-05-07 12:46:05 +02:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-06-11 05:26:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-09 15:52:27 +00:00
										 |  |  |     /* Remember pointer to module init function. */ | 
					
						
							|  |  |  |     def = PyModule_GetDef(m); | 
					
						
							| 
									
										
										
										
											2013-07-11 13:02:30 +02:00
										 |  |  |     if (def == NULL) { | 
					
						
							|  |  |  |         PyErr_Format(PyExc_SystemError, | 
					
						
							|  |  |  |                      "initialization of %s did not return an extension " | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |                      "module", name_buf); | 
					
						
							| 
									
										
										
										
											2013-07-11 11:22:21 +02:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2013-07-11 13:02:30 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     def->m_base.m_init = p0; | 
					
						
							| 
									
										
										
										
											2008-06-11 05:26:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-14 12:18:12 -06:00
										 |  |  |     PyObject *modules = PyImport_GetModuleDict(); | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |     if (_PyImport_FixupExtensionObject(m, name_unicode, filename, modules) < 0) | 
					
						
							| 
									
										
										
										
											2011-05-07 12:46:05 +02:00
										 |  |  |         goto error; | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Py_DECREF(name_unicode); | 
					
						
							|  |  |  |     Py_DECREF(name); | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |     Py_DECREF(filename); | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-14 15:54:07 -04:00
										 |  |  |     return m; | 
					
						
							| 
									
										
										
										
											2011-05-07 12:46:05 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | error: | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     Py_DECREF(name_unicode); | 
					
						
							|  |  |  |     Py_XDECREF(name); | 
					
						
							| 
									
										
										
										
											2024-04-23 08:25:50 -06:00
										 |  |  |     Py_XDECREF(filename); | 
					
						
							| 
									
										
										
										
											2011-05-07 12:46:05 +02:00
										 |  |  |     Py_XDECREF(m); | 
					
						
							|  |  |  |     return NULL; | 
					
						
							| 
									
										
										
										
											1998-08-04 22:46:29 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1999-12-22 14:09:35 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #endif /* HAVE_DYNAMIC_LOADING */
 |