116 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			116 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|   (c) copyright 1988 by the Vrije Universiteit, Amsterdam, The Netherlands.
 | |
|   See the copyright notice in the ACK home directory, in the file "Copyright".
 | |
| */
 | |
| 
 | |
| /*
 | |
|   Module:	SYSTEM
 | |
|   Author:	Ceriel J.H. Jacobs
 | |
|   Version:	$Header$
 | |
| */
 | |
| 
 | |
| /*
 | |
| 	An implementation of the Modula-2 NEWPROCESS and TRANSFER facilities
 | |
| 	using the topsize, topsave, and topload facilities.
 | |
| 	For each coroutine, a proc structure is built. For the main routine,
 | |
| 	a static space is declared to save its stack. For the other coroutines,
 | |
| 	the user specifies this space.
 | |
| */
 | |
| 
 | |
| #include <m2_traps.h>
 | |
| 
 | |
| #define MAXMAIN	2048
 | |
| 
 | |
| struct proc {
 | |
| 	unsigned size;		/* size of saved stackframe(s) */
 | |
| 	int (*proc)();		/* address of coroutine procedure */
 | |
| 	char *brk;		/* stack break of this coroutine */
 | |
| };
 | |
| 
 | |
| extern unsigned topsize();
 | |
| 
 | |
| static struct proc mainproc[MAXMAIN/sizeof(struct proc) + 1];
 | |
| 
 | |
| static struct proc *curproc = 0;/* current coroutine */
 | |
| extern char *MainLB;		/* stack break of main routine */
 | |
| 
 | |
| _SYSTEM__NEWPROCESS(p, a, n, p1)
 | |
| 	int (*p)();		/* coroutine procedure */
 | |
| 	struct proc *a;		/* pointer to area for saved stack-frame */
 | |
| 	unsigned n;		/* size of this area */
 | |
| 	struct proc **p1;	/* where to leave coroutine descriptor,
 | |
| 				   in this implementation the address of
 | |
| 				   the area for saved stack-frame(s) */
 | |
| {
 | |
| 	/*	This procedure creates a new coroutine, but does not
 | |
| 		transfer control to it. The routine "topsize" will compute the
 | |
| 		stack break, which will be the local base of this routine.
 | |
| 		Notice that we can do this because we do not need the stack
 | |
| 		above this point for this coroutine. In Modula-2, coroutines
 | |
| 		must be level 0 procedures without parameters.
 | |
| 	*/
 | |
| 	char *brk = 0;
 | |
| 	unsigned sz = topsize(&brk);
 | |
| 
 | |
| 	if (sz + sizeof(struct proc) > n) {
 | |
| 		/* not enough space */
 | |
| 		TRP(M2_TOOLARGE);
 | |
| 	}
 | |
| 	a->size = n;
 | |
| 	a->proc = p;
 | |
| 	a->brk = brk;
 | |
| 	*p1 = a;
 | |
| 	if (topsave(brk, a+1))
 | |
| 		/* stack frame saved; now just return */
 | |
| 		;
 | |
| 	else {
 | |
| 		/* We get here through the first transfer to the coroutine
 | |
| 		   created above.
 | |
| 		   This also means that curproc is now set to this coroutine.
 | |
| 		   We cannot trust the parameters anymore.
 | |
| 		   Just call the coroutine procedure.
 | |
| 		*/
 | |
| 		(*(curproc->proc))();
 | |
| 		_cleanup();
 | |
| 		_exit(0);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| _SYSTEM__TRANSFER(a, b)
 | |
| 	struct proc **a, **b;
 | |
| {
 | |
| 	/*	transfer from one coroutine to another, saving the current
 | |
| 		descriptor in the space indicated by "a", and transfering to
 | |
| 		the coroutine in descriptor "b".
 | |
| 	*/
 | |
| 	unsigned size;
 | |
| 
 | |
| 	if (! curproc) {
 | |
| 		/* the current coroutine is the main process;
 | |
| 		   initialize a coroutine descriptor for it ...
 | |
| 		*/
 | |
| 		mainproc[0].brk = MainLB;
 | |
| 		mainproc[0].size = sizeof(mainproc);
 | |
| 		curproc = &mainproc[0];
 | |
| 	}
 | |
| 	*a = curproc;		/* save current descriptor in "a" */
 | |
| 	if (*b == curproc) {
 | |
| 		/* transfer to itself is a no-op */
 | |
| 		return;
 | |
| 	}
 | |
| 	size = topsize(&(curproc->brk));
 | |
| 	if (size + sizeof(struct proc) > curproc->size) {
 | |
| 		TRP(M2_TOOLARGE);
 | |
| 	}
 | |
| 	if (topsave(curproc->brk, curproc+1)) {
 | |
| 		/* stack top saved. Now restore context of target
 | |
| 		   coroutine
 | |
| 		*/
 | |
| 		curproc = *b;
 | |
| 		topload(curproc+1);
 | |
| 		/* we never get here ... */
 | |
| 	}
 | |
| 	/* but we do get here, when a transfer is done to the coroutine in "a".
 | |
| 	*/
 | |
| }
 | 
