| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | /***********************************************************
 | 
					
						
							| 
									
										
										
										
											1995-01-04 19:08:09 +00:00
										 |  |  | Copyright 1991-1995 by Stichting Mathematisch Centrum, Amsterdam, | 
					
						
							|  |  |  | The Netherlands. | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                         All Rights Reserved | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-10-25 14:44:06 +00:00
										 |  |  | Permission to use, copy, modify, and distribute this software and its | 
					
						
							|  |  |  | documentation for any purpose and without fee is hereby granted, | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | provided that the above copyright notice appear in all copies and that | 
					
						
							| 
									
										
										
										
											1996-10-25 14:44:06 +00:00
										 |  |  | both that copyright notice and this permission notice appear in | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | supporting documentation, and that the names of Stichting Mathematisch | 
					
						
							| 
									
										
										
										
											1996-10-25 14:44:06 +00:00
										 |  |  | Centrum or CWI or Corporation for National Research Initiatives or | 
					
						
							|  |  |  | CNRI not be used in advertising or publicity pertaining to | 
					
						
							|  |  |  | distribution of the software without specific, written prior | 
					
						
							|  |  |  | permission. | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-10-25 14:44:06 +00:00
										 |  |  | While CWI is the initial source for this software, a modified version | 
					
						
							|  |  |  | is made available by the Corporation for National Research Initiatives | 
					
						
							|  |  |  | (CNRI) at the Internet address ftp://ftp.python.org.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | STICHTING MATHEMATISCH CENTRUM AND CNRI DISCLAIM ALL WARRANTIES WITH | 
					
						
							|  |  |  | REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF | 
					
						
							|  |  |  | MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH | 
					
						
							|  |  |  | CENTRUM OR CNRI 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. | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | ******************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-02-18 21:53:32 +00:00
										 |  |  | /* Readline interface for tokenizer.c and [raw_]input() in bltinmodule.c.
 | 
					
						
							|  |  |  |    By default, or when stdin is not a tty device, we have a super | 
					
						
							|  |  |  |    simple my_readline function using fgets. | 
					
						
							|  |  |  |    Optionally, we can use the GNU readline library. | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  |    my_readline() has a different return value from GNU readline(): | 
					
						
							|  |  |  |    - NULL if an interrupt occurred or if an error occurred | 
					
						
							|  |  |  |    - a malloc'ed empty string if EOF was read | 
					
						
							|  |  |  |    - a malloc'ed string ending in \n normally | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-02-18 21:53:32 +00:00
										 |  |  | #define Py_USE_NEW_NAMES 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | #include "config.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | #include "myproto.h"
 | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | #include "mymalloc.h"
 | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | #include "intrcheck.h"
 | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-02-18 21:53:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | #ifdef WITH_READLINE
 | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | extern char *readline(); | 
					
						
							| 
									
										
										
										
											1996-09-13 04:09:26 +00:00
										 |  |  | extern int rl_initialize(); | 
					
						
							|  |  |  | extern int rl_insert(); | 
					
						
							| 
									
										
										
										
											1996-12-02 18:27:33 +00:00
										 |  |  | extern int rl_bind_key(); | 
					
						
							|  |  |  | extern void add_history(); | 
					
						
							| 
									
										
										
										
											1996-09-13 04:09:26 +00:00
										 |  |  | extern char *rl_readline_name; | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <setjmp.h>
 | 
					
						
							|  |  |  | #include <signal.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static jmp_buf jbuf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ARGSUSED */ | 
					
						
							|  |  |  | static RETSIGTYPE | 
					
						
							|  |  |  | onintr(sig) | 
					
						
							|  |  |  | 	int sig; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	longjmp(jbuf, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-04-09 02:45:31 +00:00
										 |  |  | void | 
					
						
							|  |  |  | PyOS_ReadlineInit() | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	static int been_here; | 
					
						
							|  |  |  | 	if (!been_here) { | 
					
						
							|  |  |  | 		/* Force rebind of TAB to insert-tab */ | 
					
						
							| 
									
										
										
										
											1996-09-13 04:09:26 +00:00
										 |  |  | 		rl_readline_name = "python"; | 
					
						
							|  |  |  | 		rl_initialize(); | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 		rl_bind_key('\t', rl_insert); | 
					
						
							|  |  |  | 		been_here++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-04-09 02:45:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char * | 
					
						
							| 
									
										
										
										
											1997-02-18 21:53:32 +00:00
										 |  |  | PyOS_GnuReadline(prompt) | 
					
						
							| 
									
										
										
										
											1996-04-09 02:45:31 +00:00
										 |  |  | 	char *prompt; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int n; | 
					
						
							|  |  |  | 	char *p; | 
					
						
							|  |  |  | 	RETSIGTYPE (*old_inthandler)(); | 
					
						
							|  |  |  | 	PyOS_ReadlineInit(); | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 	old_inthandler = signal(SIGINT, onintr); | 
					
						
							|  |  |  | 	if (setjmp(jbuf)) { | 
					
						
							| 
									
										
										
										
											1996-01-12 01:30:55 +00:00
										 |  |  | #ifdef HAVE_SIGRELSE
 | 
					
						
							|  |  |  | 		/* This seems necessary on SunOS 4.1 (Rasmus Hahn) */ | 
					
						
							|  |  |  | 		sigrelse(SIGINT); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 		signal(SIGINT, old_inthandler); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	p = readline(prompt); | 
					
						
							|  |  |  | 	signal(SIGINT, old_inthandler); | 
					
						
							|  |  |  | 	if (p == NULL) { | 
					
						
							|  |  |  | 		p = malloc(1); | 
					
						
							|  |  |  | 		if (p != NULL) | 
					
						
							|  |  |  | 			*p = '\0'; | 
					
						
							|  |  |  | 		return p; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	n = strlen(p); | 
					
						
							|  |  |  | 	if (n > 0) | 
					
						
							|  |  |  | 		add_history(p); | 
					
						
							|  |  |  | 	if ((p = realloc(p, n+2)) != NULL) { | 
					
						
							|  |  |  | 		p[n] = '\n'; | 
					
						
							|  |  |  | 		p[n+1] = '\0'; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return p; | 
					
						
							| 
									
										
										
										
											1997-02-18 21:53:32 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* !WITH_READLINE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* This function restarts a fgets() after an EINTR error occurred
 | 
					
						
							|  |  |  |    except if PyOS_InterruptOccurred() returns true. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | my_fgets(buf, len, fp) | 
					
						
							|  |  |  | 	char *buf; | 
					
						
							|  |  |  | 	int len; | 
					
						
							|  |  |  | 	FILE *fp; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char *p; | 
					
						
							|  |  |  | 	for (;;) { | 
					
						
							|  |  |  | 		errno = 0; | 
					
						
							|  |  |  | 		p = fgets(buf, len, fp); | 
					
						
							|  |  |  | 		if (p != NULL) | 
					
						
							|  |  |  | 			return 0; /* No error */ | 
					
						
							|  |  |  | 		if (feof(fp)) { | 
					
						
							|  |  |  | 			return -1; /* EOF */ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | #ifdef EINTR
 | 
					
						
							|  |  |  | 		if (errno == EINTR) { | 
					
						
							|  |  |  | 			if (PyOS_InterruptOccurred()) { | 
					
						
							|  |  |  | 				return 1; /* Interrupt */ | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 		if (PyOS_InterruptOccurred()) { | 
					
						
							|  |  |  | 			return 1; /* Interrupt */ | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return -2; /* Error */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	/* NOTREACHED */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Readline implementation using fgets() */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | PyOS_StdioReadline(prompt) | 
					
						
							|  |  |  | 	char *prompt; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int n; | 
					
						
							|  |  |  | 	char *p; | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 	n = 100; | 
					
						
							|  |  |  | 	if ((p = malloc(n)) == NULL) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1996-01-12 01:30:55 +00:00
										 |  |  | 	fflush(stdout); | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 	if (prompt) | 
					
						
							|  |  |  | 		fprintf(stderr, "%s", prompt); | 
					
						
							| 
									
										
										
										
											1996-01-12 01:30:55 +00:00
										 |  |  | 	fflush(stderr); | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | 	switch (my_fgets(p, n, stdin)) { | 
					
						
							|  |  |  | 	case 0: /* Normal case */ | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	case 1: /* Interrupt */ | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 		free(p); | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | 	case -1: /* EOF */ | 
					
						
							|  |  |  | 	case -2: /* Error */ | 
					
						
							|  |  |  | 	default: /* Shouldn't happen */ | 
					
						
							|  |  |  | 		*p = '\0'; | 
					
						
							|  |  |  | 		break; | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | #ifdef MPW
 | 
					
						
							|  |  |  | 	/* Hack for MPW C where the prompt comes right back in the input */ | 
					
						
							|  |  |  | 	/* XXX (Actually this would be rather nice on most systems...) */ | 
					
						
							|  |  |  | 	n = strlen(prompt); | 
					
						
							|  |  |  | 	if (strncmp(p, prompt, n) == 0) | 
					
						
							|  |  |  | 		memmove(p, p + n, strlen(p) - n + 1); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 	n = strlen(p); | 
					
						
							|  |  |  | 	while (n > 0 && p[n-1] != '\n') { | 
					
						
							|  |  |  | 		int incr = n+2; | 
					
						
							|  |  |  | 		p = realloc(p, n + incr); | 
					
						
							|  |  |  | 		if (p == NULL) | 
					
						
							|  |  |  | 			return NULL; | 
					
						
							| 
									
										
										
										
											1994-08-01 11:34:53 +00:00
										 |  |  | 		if (my_fgets(p+n, incr, stdin) != 0) | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		n += strlen(p+n); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return realloc(p, n+1); | 
					
						
							| 
									
										
										
										
											1997-02-18 21:53:32 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* By initializing this function pointer, systems embedding Python can
 | 
					
						
							|  |  |  |    override the readline function. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char *(*PyOS_ReadlineFunctionPointer) Py_PROTO((char *)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Interface used by tokenizer.c and bltinmodule.c */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | PyOS_Readline(prompt) | 
					
						
							|  |  |  | 	char *prompt; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (PyOS_ReadlineFunctionPointer == NULL) { | 
					
						
							|  |  |  | #ifdef WITH_READLINE
 | 
					
						
							|  |  |  | 		if (isatty(fileno(stdin))) | 
					
						
							|  |  |  | 			PyOS_ReadlineFunctionPointer = PyOS_GnuReadline; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 			PyOS_ReadlineFunctionPointer = PyOS_StdioReadline; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return (*PyOS_ReadlineFunctionPointer)(prompt); | 
					
						
							| 
									
										
										
										
											1993-12-24 10:36:57 +00:00
										 |  |  | } |