ftp.nice.ch/pub/next/unix/communication/TipTop-goodies.s.tar.gz#/TipTop-goodies-src/expect-4.8/exp_tk.c

This is exp_tk.c in view mode; [Download] [Up]

/* exp_tk.c - Tk event interface for Expect

Written by: Don Libes, NIST, 2/6/90

Design and implementation of this program was paid for by U.S. tax
dollars.  Therefore it is public domain.  However, the author and NIST
would appreciate credit if this program or parts of it are used.

*/

/* Notes:
I'm only a little worried because Tk does not check for errno == EBADF
after calling select.  I imagine that if the user passes in a bad file
descriptor, we'll never get called back, and thus, we'll hang forever
- it would be better to at least issue a diagnostic to the user.

Another possible problem: Tk does not do file callbacks round-robin.

Another possible problem: Calling Create/DeleteFileHandler
before/after every Tcl_Eval... in expect/interact could be very
expensive.

*/


#include "exp_conf.h"
#include <stdio.h>
#include <errno.h>

#ifdef HAVE_PTYTRAP
#  include <sys/ptyio.h>
#endif

#include "tk.h"

#include "exp_global.h"
#include "exp_command.h"	/* for struct f defs */
#include "exp_event.h"

/* Tk_DoOneEvent will call our filehandler which will set the following vars */
/* enabling us to know where and what kind of I/O we can do */
/*#define EXP_SPAWN_ID_BAD	-1*/
/*#define EXP_SPAWN_ID_TIMEOUT	-2*/	/* really indicates a timeout */

static int ready_fd = EXP_SPAWN_ID_BAD;
static int ready_mask;
static int default_mask = TK_READABLE | TK_EXCEPTION;
#if 0
#ifdef HAVE_PTYTRAP
static int default_mask = TK_READABLE | TK_EXCEPTION;
#else
static int default_mask = TK_READABLE;
#endif
#endif

#if 1
/* reduce calls to malloc/free inside Tk_...FileHandler */
#define Fast_Tk_DeleteFileHandler(fd) Tk_CreateFileHandler(fd,0,exp_filehandler,(ClientData)0)
#endif /* 1 */

static void exp_filehandler(clientData,mask)
ClientData clientData;
int mask;
{
	if (ready_fd == (int)clientData) {
		Fast_Tk_DeleteFileHandler(ready_fd);
		fs[ready_fd].armed = FALSE;
	} else {
		ready_fd = (int)clientData;
		ready_mask = mask;
	}
}

/*ARGSUSED*/
static void
exp_timehandler(clientData)
ClientData clientData;
{
	/* clientData == &timer_fired */
	*(int *)clientData = TRUE;	

	/* old implementation */
	/* ready_fd = EXP_SPAWN_ID_TIMEOUT; */
}

#if 0
void
exp_disarm_event_handler(count,fds)
int count;
int *fds;
{
	int i;

	for (i=0;i<count;i++) {
		Tk_DeleteFileHandler(fds[i]);
	}
}


static void
exp_arm_event_handlers(count,fds)
int count;
int *fds;
{
	int i;
	int flag = TK_WRITABLE;

	flag |= TK_EXCEPTION;
#if 0
#ifdef HAVE_PTYTRAP
	flag |= TK_EXCEPTION;
#endif
#endif

	for (i=0;i<count;i++) {
		Tk_CreateFileHandler(fds[i],flag,exp_filehandler,(ClientData)fds[i]);
	}
}
#endif

void
exp_event_disarm(fd)
int fd;
{
	Tk_DeleteFileHandler(fd);
}

/* returns status, one of EOF, TIMEOUT, ERROR or DATA */
/*ARGSUSED*/
int exp_get_next_event(interp,masters, n,master_out,timeout,key)
Tcl_Interp *interp;
int *masters;
int n;			/* # of masters */
int *master_out;	/* 1st ready master, not set if none */
int timeout;		/* seconds */
int key;
{
	static rr = 0;	/* round robin ptr */
	int i;	/* index into in-array */

	int timer_created = FALSE;
	int timer_fired = FALSE;
	Tk_TimerToken timetoken;/* handle to Tk timehandler descriptor */

	for (;;) {
		int m;
		struct f *f;

		/* if anything has been touched by someone else, report that */
		/* an event has been received */

		for (i=0;i<n;i++) {
			rr++;
			if (rr >= n) rr = 0;

			m = masters[rr];
			f = fs + m;

			if (f->key != key) {
				f->key = key;
				f->force_read = FALSE;
				*master_out = m;
				return(EXP_DATA_OLD);
			} else if ((!f->force_read) && (f->size != 0)) {
				*master_out = m;
				return(EXP_DATA_OLD);
			}
		}

		if (!timer_created) {
			if (timeout >= 0) {
				timetoken = Tk_CreateTimerHandler(1000*timeout,
						exp_timehandler,
						(ClientData)&timer_fired);
				timer_created = TRUE;
			}
		}

		for (;;) {
			extern int fd_max;
			int j;

			/* make sure that all fds that should be armed are */
			for (j=0;j<n;j++) {
				int k = masters[j];

				if (!fs[k].armed) {
					Tk_CreateFileHandler(k,default_mask,
							     exp_filehandler,
							     (ClientData)k);
					fs[k].armed = TRUE;
				}
			}

			Tk_DoOneEvent(0);	/* do any event */
			if (timer_fired) return(EXP_TIMEOUT);
			if (ready_fd == EXP_SPAWN_ID_BAD) continue;

			/* if it was from something we're not looking for at */
			/* the moment, ignore it */
			for (j=0;j<n;j++) {
				if (ready_fd == masters[j]) goto found;
			}
			/* not found */
			Fast_Tk_DeleteFileHandler(ready_fd);
			fs[ready_fd].armed = FALSE;
			ready_fd = EXP_SPAWN_ID_BAD;
			continue;
		found:
			*master_out = ready_fd;
			ready_fd = EXP_SPAWN_ID_BAD;

			if (timer_created) Tk_DeleteTimerHandler(timetoken);

/*#ifdef HAVE_PTYTRAP*/
			if (ready_mask & TK_EXCEPTION) {
#ifndef HAVE_PTYTRAP
				return(EXP_EOF);
#else
				struct request_info ioctl_info;
				if (ioctl(*master_out,TIOCREQCHECK,&ioctl_info) < 0) {
					exp_debuglog("ioctl error on TIOCREQCHECK: %d", errno);
					return(TCL_ERROR);
				}
				if (ioctl_info.request == TIOCCLOSE) {
					return(EXP_EOF);
				}
				if (ioctl(*master_out, TIOCREQSET, &ioctl_info) < 0) {
					exp_debuglog("ioctl error on TIOCREQSET after ioctl or open on slave: %d", errno);
				}
				/* presumably, we trapped an open here */
				continue;
#endif
			}
/*#endif *//*HAVE_PTYTRAP*/
			return(EXP_DATA_NEW);
		}
	}
}

void
exp_usleep(usec)
long usec;
{
	int timer_fired = FALSE;

	Tk_CreateTimerHandler(usec/1000,exp_timehandler,(ClientData)&timer_fired);

	while (1) {
		Tk_DoOneEvent(0);
		if (timer_fired) return;

		if (ready_fd == EXP_SPAWN_ID_BAD) continue;

		Fast_Tk_DeleteFileHandler(ready_fd);
		fs[ready_fd].armed = FALSE;
		ready_fd = EXP_SPAWN_ID_BAD;
	}
}

static void
exp_event_exit_real(interp)
Tcl_Interp *interp;
{
#if TCL_MAJOR_VERSION
	Tcl_Eval(interp,"destroy .",0,(char **)NULL);
#else
	Tcl_Eval(interp,"destroy .");
#endif
}

/* set things up for later calls to event handler */
void
exp_init_event()
{
	exp_event_exit = exp_event_exit_real;
}

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.