/src/cpython/Python/dynload_shlib.c
Line | Count | Source (jump to first uncovered line) |
1 | | |
2 | | /* Support for dynamic loading of extension modules */ |
3 | | |
4 | | #include "Python.h" |
5 | | #include "pycore_fileutils.h" // struct _Py_stat_struct |
6 | | #include "pycore_import.h" // _PyImport_GetDLOpenFlags() |
7 | | #include "pycore_importdl.h" |
8 | | #include "pycore_interp.h" // _PyInterpreterState.dlopenflags |
9 | | #include "pycore_pystate.h" // _PyInterpreterState_GET() |
10 | | |
11 | | #include <sys/types.h> |
12 | | #include <sys/stat.h> |
13 | | |
14 | | #if defined(__NetBSD__) |
15 | | #include <sys/param.h> |
16 | | #if (NetBSD < 199712) |
17 | | #include <nlist.h> |
18 | | #include <link.h> |
19 | | #define dlerror() "error in dynamic linking" |
20 | | #endif |
21 | | #endif /* NetBSD */ |
22 | | |
23 | | #ifdef HAVE_DLFCN_H |
24 | | #include <dlfcn.h> |
25 | | #endif |
26 | | |
27 | | #if (defined(__OpenBSD__) || defined(__NetBSD__)) && !defined(__ELF__) |
28 | | #define LEAD_UNDERSCORE "_" |
29 | | #else |
30 | 79 | #define LEAD_UNDERSCORE "" |
31 | | #endif |
32 | | |
33 | | /* The .so extension module ABI tag, supplied by the Makefile via |
34 | | Makefile.pre.in and configure. This is used to discriminate between |
35 | | incompatible .so files so that extensions for different Python builds can |
36 | | live in the same directory. E.g. foomodule.cpython-32.so |
37 | | */ |
38 | | |
39 | | const char *_PyImport_DynLoadFiletab[] = { |
40 | | #ifdef __CYGWIN__ |
41 | | ".dll", |
42 | | #else /* !__CYGWIN__ */ |
43 | | "." SOABI ".so", |
44 | | #ifdef ALT_SOABI |
45 | | "." ALT_SOABI ".so", |
46 | | #endif |
47 | | ".abi" PYTHON_ABI_STRING ".so", |
48 | | ".so", |
49 | | #endif /* __CYGWIN__ */ |
50 | | NULL, |
51 | | }; |
52 | | |
53 | | |
54 | | dl_funcptr |
55 | | _PyImport_FindSharedFuncptr(const char *prefix, |
56 | | const char *shortname, |
57 | | const char *pathname, FILE *fp) |
58 | 79 | { |
59 | 79 | dl_funcptr p; |
60 | 79 | void *handle; |
61 | 79 | char funcname[258]; |
62 | 79 | char pathbuf[260]; |
63 | 79 | int dlopenflags=0; |
64 | | |
65 | 79 | if (strchr(pathname, '/') == NULL) { |
66 | | /* Prefix bare filename with "./" */ |
67 | 0 | PyOS_snprintf(pathbuf, sizeof(pathbuf), "./%-.255s", pathname); |
68 | 0 | pathname = pathbuf; |
69 | 0 | } |
70 | | |
71 | 79 | PyOS_snprintf(funcname, sizeof(funcname), |
72 | 79 | LEAD_UNDERSCORE "%.20s_%.200s", prefix, shortname); |
73 | | |
74 | 79 | if (fp != NULL) { |
75 | 0 | struct _Py_stat_struct status; |
76 | 0 | if (_Py_fstat(fileno(fp), &status) == -1) |
77 | 0 | return NULL; |
78 | 0 | } |
79 | | |
80 | 79 | dlopenflags = _PyImport_GetDLOpenFlags(_PyInterpreterState_GET()); |
81 | | |
82 | 79 | handle = dlopen(pathname, dlopenflags); |
83 | | |
84 | 79 | if (handle == NULL) { |
85 | 0 | PyObject *mod_name; |
86 | 0 | PyObject *path; |
87 | 0 | PyObject *error_ob; |
88 | 0 | const char *error = dlerror(); |
89 | 0 | if (error == NULL) |
90 | 0 | error = "unknown dlopen() error"; |
91 | 0 | error_ob = PyUnicode_DecodeLocale(error, "surrogateescape"); |
92 | 0 | if (error_ob == NULL) |
93 | 0 | return NULL; |
94 | 0 | mod_name = PyUnicode_FromString(shortname); |
95 | 0 | if (mod_name == NULL) { |
96 | 0 | Py_DECREF(error_ob); |
97 | 0 | return NULL; |
98 | 0 | } |
99 | 0 | path = PyUnicode_DecodeFSDefault(pathname); |
100 | 0 | if (path == NULL) { |
101 | 0 | Py_DECREF(error_ob); |
102 | 0 | Py_DECREF(mod_name); |
103 | 0 | return NULL; |
104 | 0 | } |
105 | 0 | PyErr_SetImportError(error_ob, mod_name, path); |
106 | 0 | Py_DECREF(error_ob); |
107 | 0 | Py_DECREF(mod_name); |
108 | 0 | Py_DECREF(path); |
109 | 0 | return NULL; |
110 | 0 | } |
111 | 79 | p = (dl_funcptr) dlsym(handle, funcname); |
112 | 79 | return p; |
113 | 79 | } |