| 
									
										
										
										
											1999-12-20 21:18:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Support for dynamic loading of extension modules */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Python.h"
 | 
					
						
							|  |  |  | #include "importdl.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <sys/types.h>
 | 
					
						
							|  |  |  | #include <sys/stat.h>
 | 
					
						
							|  |  |  | #if defined(__NetBSD__) && (NetBSD < 199712)
 | 
					
						
							|  |  |  | #include <nlist.h>
 | 
					
						
							|  |  |  | #include <link.h>
 | 
					
						
							|  |  |  | #define dlerror() "error in dynamic linking"
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #ifdef HAVE_DLFCN_H
 | 
					
						
							|  |  |  | #include <dlfcn.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-10-25 22:07:45 +00:00
										 |  |  | #ifdef __OpenBSD__
 | 
					
						
							|  |  |  | #define LEAD_UNDERSCORE "_"
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #define LEAD_UNDERSCORE ""
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-20 21:18:49 +00:00
										 |  |  | #ifndef RTLD_LAZY
 | 
					
						
							|  |  |  | #define RTLD_LAZY 1
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const struct filedescr _PyImport_DynLoadFiletab[] = { | 
					
						
							| 
									
										
										
										
											2000-10-05 19:24:26 +00:00
										 |  |  | #ifdef __CYGWIN__
 | 
					
						
							|  |  |  | 	{".dll", "rb", C_EXTENSION}, | 
					
						
							| 
									
										
										
										
											2001-01-10 21:17:27 +00:00
										 |  |  | 	{"module.dll", "rb", C_EXTENSION}, | 
					
						
							| 
									
										
										
										
											2000-10-05 19:24:26 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											1999-12-20 21:18:49 +00:00
										 |  |  | 	{".so", "rb", C_EXTENSION}, | 
					
						
							|  |  |  | 	{"module.so", "rb", C_EXTENSION}, | 
					
						
							| 
									
										
										
										
											2000-10-05 19:24:26 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1999-12-20 21:18:49 +00:00
										 |  |  | 	{0, 0} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct { | 
					
						
							|  |  |  | 	dev_t dev; | 
					
						
							|  |  |  | 	ino_t ino; | 
					
						
							|  |  |  | 	void *handle; | 
					
						
							|  |  |  | } handles[128]; | 
					
						
							|  |  |  | static int nhandles = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-22 14:09:35 +00:00
										 |  |  | dl_funcptr _PyImport_GetDynLoadFunc(const char *fqname, const char *shortname, | 
					
						
							| 
									
										
										
										
											1999-12-20 21:18:49 +00:00
										 |  |  | 				    const char *pathname, FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	dl_funcptr p; | 
					
						
							|  |  |  | 	void *handle; | 
					
						
							| 
									
										
										
										
											1999-12-22 14:09:35 +00:00
										 |  |  | 	char funcname[258]; | 
					
						
							|  |  |  | 	char pathbuf[260]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (strchr(pathname, '/') == NULL) { | 
					
						
							|  |  |  | 		/* Prefix bare filename with "./" */ | 
					
						
							|  |  |  | 		sprintf(pathbuf, "./%-.255s", pathname); | 
					
						
							|  |  |  | 		pathname = pathbuf; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-10-25 22:07:45 +00:00
										 |  |  | 	sprintf(funcname, LEAD_UNDERSCORE "init%.200s", shortname); | 
					
						
							| 
									
										
										
										
											1999-12-20 21:18:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (fp != NULL) { | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 		struct stat statb; | 
					
						
							|  |  |  | 		fstat(fileno(fp), &statb); | 
					
						
							|  |  |  | 		for (i = 0; i < nhandles; i++) { | 
					
						
							|  |  |  | 			if (statb.st_dev == handles[i].dev && | 
					
						
							|  |  |  | 			    statb.st_ino == handles[i].ino) { | 
					
						
							|  |  |  | 				p = (dl_funcptr) dlsym(handles[i].handle, | 
					
						
							|  |  |  | 						       funcname); | 
					
						
							|  |  |  | 				return p; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		if (nhandles < 128) { | 
					
						
							|  |  |  | 			handles[nhandles].dev = statb.st_dev; | 
					
						
							|  |  |  | 			handles[nhandles].ino = statb.st_ino; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef RTLD_NOW
 | 
					
						
							|  |  |  | 	/* RTLD_NOW: resolve externals now
 | 
					
						
							|  |  |  | 	   (i.e. core dump now if some are missing) */ | 
					
						
							|  |  |  | 	handle = dlopen(pathname, RTLD_NOW); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	if (Py_VerboseFlag) | 
					
						
							|  |  |  | 		printf("dlopen(\"%s\", %d);\n", pathname, | 
					
						
							|  |  |  | 		       RTLD_LAZY); | 
					
						
							|  |  |  | 	handle = dlopen(pathname, RTLD_LAZY); | 
					
						
							|  |  |  | #endif /* RTLD_NOW */
 | 
					
						
							|  |  |  | 	if (handle == NULL) { | 
					
						
							|  |  |  | 		PyErr_SetString(PyExc_ImportError, dlerror()); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (fp != NULL && nhandles < 128) | 
					
						
							|  |  |  | 		handles[nhandles++].handle = handle; | 
					
						
							|  |  |  | 	p = (dl_funcptr) dlsym(handle, funcname); | 
					
						
							|  |  |  | 	return p; | 
					
						
							|  |  |  | } |