| 
									
										
										
										
											1991-02-19 12:39:46 +00:00
										 |  |  | /***********************************************************
 | 
					
						
							| 
									
										
										
										
											1992-04-05 14:26:55 +00:00
										 |  |  | Copyright 1991, 1992 by Stichting Mathematisch Centrum, Amsterdam, The | 
					
						
							| 
									
										
										
										
											1991-02-19 12:39:46 +00:00
										 |  |  | 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-12-20 15:06:42 +00:00
										 |  |  | /* Traceback implementation */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "allobjects.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1992-02-26 15:17:59 +00:00
										 |  |  | #include "sysmodule.h"
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | #include "compile.h"
 | 
					
						
							|  |  |  | #include "frameobject.h"
 | 
					
						
							|  |  |  | #include "traceback.h"
 | 
					
						
							|  |  |  | #include "structmember.h"
 | 
					
						
							| 
									
										
										
										
											1992-02-26 15:17:59 +00:00
										 |  |  | #include "osdefs.h"
 | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | typedef struct _tracebackobject { | 
					
						
							|  |  |  | 	OB_HEAD | 
					
						
							|  |  |  | 	struct _tracebackobject *tb_next; | 
					
						
							|  |  |  | 	frameobject *tb_frame; | 
					
						
							|  |  |  | 	int tb_lasti; | 
					
						
							|  |  |  | 	int tb_lineno; | 
					
						
							|  |  |  | } tracebackobject; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define OFF(x) offsetof(tracebackobject, x)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct memberlist tb_memberlist[] = { | 
					
						
							|  |  |  | 	{"tb_next",	T_OBJECT,	OFF(tb_next)}, | 
					
						
							|  |  |  | 	{"tb_frame",	T_OBJECT,	OFF(tb_frame)}, | 
					
						
							|  |  |  | 	{"tb_lasti",	T_INT,		OFF(tb_lasti)}, | 
					
						
							|  |  |  | 	{"tb_lineno",	T_INT,		OFF(tb_lineno)}, | 
					
						
							|  |  |  | 	{NULL}	/* Sentinel */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static object * | 
					
						
							|  |  |  | tb_getattr(tb, name) | 
					
						
							|  |  |  | 	tracebackobject *tb; | 
					
						
							|  |  |  | 	char *name; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return getmember((char *)tb, tb_memberlist, name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | tb_dealloc(tb) | 
					
						
							|  |  |  | 	tracebackobject *tb; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	XDECREF(tb->tb_next); | 
					
						
							|  |  |  | 	XDECREF(tb->tb_frame); | 
					
						
							|  |  |  | 	DEL(tb); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static typeobject Tracebacktype = { | 
					
						
							|  |  |  | 	OB_HEAD_INIT(&Typetype) | 
					
						
							|  |  |  | 	0, | 
					
						
							|  |  |  | 	"traceback", | 
					
						
							|  |  |  | 	sizeof(tracebackobject), | 
					
						
							|  |  |  | 	0, | 
					
						
							|  |  |  | 	tb_dealloc,	/*tp_dealloc*/ | 
					
						
							|  |  |  | 	0,		/*tp_print*/ | 
					
						
							|  |  |  | 	tb_getattr,	/*tp_getattr*/ | 
					
						
							|  |  |  | 	0,		/*tp_setattr*/ | 
					
						
							|  |  |  | 	0,		/*tp_compare*/ | 
					
						
							|  |  |  | 	0,		/*tp_repr*/ | 
					
						
							|  |  |  | 	0,		/*tp_as_number*/ | 
					
						
							|  |  |  | 	0,		/*tp_as_sequence*/ | 
					
						
							|  |  |  | 	0,		/*tp_as_mapping*/ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define is_tracebackobject(v) ((v)->ob_type == &Tracebacktype)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static tracebackobject * | 
					
						
							|  |  |  | newtracebackobject(next, frame, lasti, lineno) | 
					
						
							|  |  |  | 	tracebackobject *next; | 
					
						
							|  |  |  | 	frameobject *frame; | 
					
						
							|  |  |  | 	int lasti, lineno; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	tracebackobject *tb; | 
					
						
							|  |  |  | 	if ((next != NULL && !is_tracebackobject(next)) || | 
					
						
							|  |  |  | 			frame == NULL || !is_frameobject(frame)) { | 
					
						
							|  |  |  | 		err_badcall(); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	tb = NEWOBJ(tracebackobject, &Tracebacktype); | 
					
						
							|  |  |  | 	if (tb != NULL) { | 
					
						
							|  |  |  | 		XINCREF(next); | 
					
						
							|  |  |  | 		tb->tb_next = next; | 
					
						
							|  |  |  | 		XINCREF(frame); | 
					
						
							|  |  |  | 		tb->tb_frame = frame; | 
					
						
							|  |  |  | 		tb->tb_lasti = lasti; | 
					
						
							|  |  |  | 		tb->tb_lineno = lineno; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return tb; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static tracebackobject *tb_current = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											1992-01-14 18:44:48 +00:00
										 |  |  | tb_here(frame) | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	frameobject *frame; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	tracebackobject *tb; | 
					
						
							| 
									
										
										
										
											1992-01-14 18:44:48 +00:00
										 |  |  | 	tb = newtracebackobject(tb_current, frame, frame->f_lasti, frame->f_lineno); | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	if (tb == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	XDECREF(tb_current); | 
					
						
							|  |  |  | 	tb_current = tb; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | object * | 
					
						
							|  |  |  | tb_fetch() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	object *v; | 
					
						
							|  |  |  | 	v = (object *)tb_current; | 
					
						
							|  |  |  | 	tb_current = NULL; | 
					
						
							|  |  |  | 	return v; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | tb_store(v) | 
					
						
							|  |  |  | 	object *v; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (v != NULL && !is_tracebackobject(v)) { | 
					
						
							|  |  |  | 		err_badcall(); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	XDECREF(tb_current); | 
					
						
							|  |  |  | 	XINCREF(v); | 
					
						
							|  |  |  | 	tb_current = (tracebackobject *)v; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | tb_displayline(fp, filename, lineno) | 
					
						
							|  |  |  | 	FILE *fp; | 
					
						
							|  |  |  | 	char *filename; | 
					
						
							|  |  |  | 	int lineno; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	FILE *xfp; | 
					
						
							| 
									
										
										
										
											1992-02-26 15:17:59 +00:00
										 |  |  | 	char linebuf[1000]; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	int i; | 
					
						
							|  |  |  | 	xfp = fopen(filename, "r"); | 
					
						
							|  |  |  | 	if (xfp == NULL) { | 
					
						
							| 
									
										
										
										
											1992-02-26 15:17:59 +00:00
										 |  |  | 		/* Search tail of filename in sys.path before giving up */ | 
					
						
							|  |  |  | 		object *path; | 
					
						
							|  |  |  | 		char *tail = strrchr(filename, SEP); | 
					
						
							|  |  |  | 		if (tail == NULL) | 
					
						
							|  |  |  | 			tail = filename; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			tail++; | 
					
						
							|  |  |  | 		path = sysget("path"); | 
					
						
							|  |  |  | 		if (path != NULL && is_listobject(path)) { | 
					
						
							|  |  |  | 			int npath = getlistsize(path); | 
					
						
							|  |  |  | 			char namebuf[MAXPATHLEN+1]; | 
					
						
							|  |  |  | 			for (i = 0; i < npath; i++) { | 
					
						
							|  |  |  | 				object *v = getlistitem(path, i); | 
					
						
							|  |  |  | 				if (is_stringobject(v)) { | 
					
						
							|  |  |  | 					int len; | 
					
						
							|  |  |  | 					strcpy(namebuf, getstringvalue(v)); | 
					
						
							|  |  |  | 					len = getstringsize(v); | 
					
						
							|  |  |  | 					if (len > 0 && namebuf[len-1] != SEP) | 
					
						
							|  |  |  | 						namebuf[len++] = SEP; | 
					
						
							|  |  |  | 					strcpy(namebuf+len, tail); | 
					
						
							|  |  |  | 					xfp = fopen(namebuf, "r"); | 
					
						
							|  |  |  | 					if (xfp != NULL) { | 
					
						
							|  |  |  | 						filename = namebuf; | 
					
						
							|  |  |  | 						break; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1992-02-26 15:17:59 +00:00
										 |  |  | 	fprintf(fp, "  File \"%s\"", filename); | 
					
						
							|  |  |  | #ifdef applec /* MPW */
 | 
					
						
							|  |  |  | 	/* This is needed by MPW's File and Line commands */ | 
					
						
							|  |  |  | 	fprintf(fp, "; "); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	/* This is needed by Emacs' compile command */ | 
					
						
							|  |  |  | 	fprintf(fp, ", "); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	fprintf(fp, "line %d\n", lineno); | 
					
						
							|  |  |  | 	if (xfp == NULL) | 
					
						
							|  |  |  | 		return; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 	for (i = 0; i < lineno; i++) { | 
					
						
							| 
									
										
										
										
											1992-02-26 15:17:59 +00:00
										 |  |  | 		if (fgets(linebuf, sizeof linebuf, xfp) == NULL) | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (i == lineno) { | 
					
						
							| 
									
										
										
										
											1992-02-26 15:17:59 +00:00
										 |  |  | 		char *p = linebuf; | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 		while (*p == ' ' || *p == '\t') | 
					
						
							|  |  |  | 			p++; | 
					
						
							|  |  |  | 		fprintf(fp, "    %s", p); | 
					
						
							|  |  |  | 		if (strchr(p, '\n') == NULL) | 
					
						
							|  |  |  | 			fprintf(fp, "\n"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	fclose(xfp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | tb_printinternal(tb, fp) | 
					
						
							|  |  |  | 	tracebackobject *tb; | 
					
						
							|  |  |  | 	FILE *fp; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1992-02-26 15:17:59 +00:00
										 |  |  | 	while (tb != NULL && !intrcheck()) { | 
					
						
							| 
									
										
										
										
											1990-12-20 15:06:42 +00:00
										 |  |  | 		tb_displayline(fp, | 
					
						
							|  |  |  | 		     getstringvalue(tb->tb_frame->f_code->co_filename), | 
					
						
							|  |  |  | 							tb->tb_lineno); | 
					
						
							|  |  |  | 		tb = tb->tb_next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | tb_print(v, fp) | 
					
						
							|  |  |  | 	object *v; | 
					
						
							|  |  |  | 	FILE *fp; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (v == NULL) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 	if (!is_tracebackobject(v)) { | 
					
						
							|  |  |  | 		err_badcall(); | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	sysset("last_traceback", v); | 
					
						
							|  |  |  | 	tb_printinternal((tracebackobject *)v, fp); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } |