| 
									
										
										
										
											1991-02-19 12:39:46 +00:00
										 |  |  | /***********************************************************
 | 
					
						
							|  |  |  | Copyright 1991 by Stichting Mathematisch Centrum, Amsterdam, The | 
					
						
							|  |  |  | Netherlands. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         All Rights Reserved | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Permission to use, copy, modify, and distribute this software and its  | 
					
						
							|  |  |  | documentation for any purpose and without fee is hereby granted,  | 
					
						
							|  |  |  | provided that the above copyright notice appear in all copies and that | 
					
						
							|  |  |  | both that copyright notice and this permission notice appear in  | 
					
						
							|  |  |  | supporting documentation, and that the names of Stichting Mathematisch | 
					
						
							|  |  |  | Centrum or CWI not be used in advertising or publicity pertaining to | 
					
						
							|  |  |  | distribution of the software without specific, written prior permission. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | STICHTING MATHEMATISCH CENTRUM DISCLAIMS ALL WARRANTIES WITH REGARD TO | 
					
						
							|  |  |  | THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND | 
					
						
							|  |  |  | FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH CENTRUM BE LIABLE | 
					
						
							|  |  |  | FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | 
					
						
							|  |  |  | WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | 
					
						
							|  |  |  | ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT | 
					
						
							|  |  |  | OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ******************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | /* Module definition and import implementation */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | #include "allobjects.h"
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "node.h"
 | 
					
						
							|  |  |  | #include "token.h"
 | 
					
						
							|  |  |  | #include "graminit.h"
 | 
					
						
							|  |  |  | #include "import.h"
 | 
					
						
							|  |  |  | #include "errcode.h"
 | 
					
						
							|  |  |  | #include "sysmodule.h"
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | #include "pythonrun.h"
 | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | /* Define pathname separator used in file names */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef THINK_C
 | 
					
						
							|  |  |  | #define SEP ':'
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef SEP
 | 
					
						
							|  |  |  | #define SEP '/'
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | static object *modules; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Initialization */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | initimport() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	if ((modules = newdictobject()) == NULL) | 
					
						
							|  |  |  | 		fatal("no mem for dictionary of modules"); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | object * | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | get_modules() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return modules; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | object * | 
					
						
							|  |  |  | add_module(name) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	char *name; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *m; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	if ((m = dictlookup(modules, name)) != NULL && is_moduleobject(m)) | 
					
						
							|  |  |  | 		return m; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	m = newmoduleobject(name); | 
					
						
							|  |  |  | 	if (m == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	if (dictinsert(modules, name, m) != 0) { | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		DECREF(m); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	DECREF(m); /* Yes, it still exists, in modules! */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	return m; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static FILE * | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | open_module(name, suffix, namebuf) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	char *name; | 
					
						
							|  |  |  | 	char *suffix; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	char *namebuf; /* XXX No buffer overflow checks! */ | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	object *path; | 
					
						
							|  |  |  | 	FILE *fp; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	path = sysget("path"); | 
					
						
							|  |  |  | 	if (path == NULL || !is_listobject(path)) { | 
					
						
							|  |  |  | 		strcpy(namebuf, name); | 
					
						
							|  |  |  | 		strcat(namebuf, suffix); | 
					
						
							|  |  |  | 		fp = fopen(namebuf, "r"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		int npath = getlistsize(path); | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 		fp = NULL; | 
					
						
							|  |  |  | 		for (i = 0; i < npath; i++) { | 
					
						
							|  |  |  | 			object *v = getlistitem(path, i); | 
					
						
							|  |  |  | 			int len; | 
					
						
							|  |  |  | 			if (!is_stringobject(v)) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			strcpy(namebuf, getstringvalue(v)); | 
					
						
							|  |  |  | 			len = getstringsize(v); | 
					
						
							|  |  |  | 			if (len > 0 && namebuf[len-1] != SEP) | 
					
						
							|  |  |  | 				namebuf[len++] = SEP; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 			strcpy(namebuf+len, name); | 
					
						
							|  |  |  | 			strcat(namebuf, suffix); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 			fp = fopen(namebuf, "r"); | 
					
						
							|  |  |  | 			if (fp != NULL) | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return fp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | get_module(m, name, m_ret) | 
					
						
							|  |  |  | 	/*module*/object *m; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	char *name; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	object **m_ret; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	object *d; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	FILE *fp; | 
					
						
							| 
									
										
										
										
											1990-10-26 14:58:58 +00:00
										 |  |  | 	node *n; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	int err; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	char namebuf[256]; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	fp = open_module(name, ".py", namebuf); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	if (fp == NULL) { | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 		if (m == NULL) | 
					
						
							|  |  |  | 			err_setstr(NameError, name); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			err_setstr(RuntimeError, "no module source file"); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	err = parse_file(fp, namebuf, file_input, &n); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	fclose(fp); | 
					
						
							|  |  |  | 	if (err != E_DONE) { | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 		err_input(err); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (m == NULL) { | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 		m = add_module(name); | 
					
						
							|  |  |  | 		if (m == NULL) { | 
					
						
							|  |  |  | 			freetree(n); | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		*m_ret = m; | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	d = getmoduledict(m); | 
					
						
							|  |  |  | 	return run_node(n, namebuf, d, d); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | load_module(name) | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *m, *v; | 
					
						
							|  |  |  | 	v = get_module((object *)NULL, name, &m); | 
					
						
							|  |  |  | 	if (v == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	DECREF(v); | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	return m; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | object * | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | import_module(name) | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	char *name; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *m; | 
					
						
							| 
									
										
										
										
											1991-02-19 12:23:57 +00:00
										 |  |  | 	if ((m = dictlookup(modules, name)) == NULL) { | 
					
						
							|  |  |  | 		if (init_builtin(name)) { | 
					
						
							|  |  |  | 			if ((m = dictlookup(modules, name)) == NULL) | 
					
						
							|  |  |  | 				err_setstr(SystemError, "builtin module missing"); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			m = load_module(name); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-10-14 12:07:46 +00:00
										 |  |  | 	return m; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1990-10-26 14:58:58 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | object * | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | reload_module(m) | 
					
						
							| 
									
										
										
										
											1990-10-26 14:58:58 +00:00
										 |  |  | 	object *m; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (m == NULL || !is_moduleobject(m)) { | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 		err_setstr(TypeError, "reload() argument must be module"); | 
					
						
							| 
									
										
										
										
											1990-10-26 14:58:58 +00:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	/* XXX Ought to check for builtin modules -- can't reload these... */ | 
					
						
							|  |  |  | 	return get_module(m, getmodulename(m), (object **)NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | cleardict(d) | 
					
						
							|  |  |  | 	object *d; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	for (i = getdictsize(d); --i >= 0; ) { | 
					
						
							|  |  |  | 		char *k; | 
					
						
							|  |  |  | 		k = getdictkey(d, i); | 
					
						
							|  |  |  | 		if (k != NULL) | 
					
						
							|  |  |  | 			(void) dictremove(d, k); | 
					
						
							| 
									
										
										
										
											1990-10-26 14:58:58 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | doneimport() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (modules != NULL) { | 
					
						
							|  |  |  | 		int i; | 
					
						
							|  |  |  | 		/* Explicitly erase all modules; this is the safest way
 | 
					
						
							|  |  |  | 		   to get rid of at least *some* circular dependencies */ | 
					
						
							|  |  |  | 		for (i = getdictsize(modules); --i >= 0; ) { | 
					
						
							|  |  |  | 			char *k; | 
					
						
							|  |  |  | 			k = getdictkey(modules, i); | 
					
						
							|  |  |  | 			if (k != NULL) { | 
					
						
							|  |  |  | 				object *m; | 
					
						
							|  |  |  | 				m = dictlookup(modules, k); | 
					
						
							|  |  |  | 				if (m != NULL && is_moduleobject(m)) { | 
					
						
							|  |  |  | 					object *d; | 
					
						
							|  |  |  | 					d = getmoduledict(m); | 
					
						
							|  |  |  | 					if (d != NULL && is_dictobject(d)) { | 
					
						
							|  |  |  | 						cleardict(d); | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		cleardict(modules); | 
					
						
							| 
									
										
										
										
											1990-10-26 14:58:58 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	DECREF(modules); | 
					
						
							| 
									
										
										
										
											1990-10-26 14:58:58 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1991-02-19 12:23:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Initialize built-in modules when first imported */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern struct { | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | 	void (*initfunc)(); | 
					
						
							|  |  |  | } inittab[]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | init_builtin(name) | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	for (i = 0; inittab[i].name != NULL; i++) { | 
					
						
							|  |  |  | 		if (strcmp(name, inittab[i].name) == 0) { | 
					
						
							|  |  |  | 			(*inittab[i].initfunc)(); | 
					
						
							|  |  |  | 			return 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } |