2.11BSD/ingres/source/parser/control.c

Compare this file to the similar file:
Show the results in this format:

# include	"../ingres.h"
# include	"../aux.h"
# include	"../pipes.h"
# include	"../symbol.h"
# include	"parser.h"
# include	"../scanner.h"

/*
**  CONTROL.C -- -- collection of control functions for the parser
**
**	These routines administrate the operation of the parser for internal
**	sequencing.  There are 3 pairs of routines, one pair for each
**	quel statement, one for each go-block, and one to handle syncronization
**	with up and down processes.
**
**	Defines:
**		init_quelst	-- initialize for a quel statement
**		endquelst	-- clean up after a quel statement
**		syncup		-- sync the process above
**		syncdn		-- sync the process below
**		startgo		-- initialize for a go-block
**		endgo		-- clean up after a go-block
**
**	Trace Flags:
**		33.0		-- in endquelst for done with it
**		60.0		-- reading tuple count and saving
**		60.1		-- final tuple count, written to monitor
**
**	History:
**		15 Jan 79 (rick)	collected and documented more
**		ancient history
*/







/*
** INIT_QUELST -- set vbles for default mode before each quel statement
*/
init_quelst()
{
	extern int	neederr();

	initp();				/* reset dbu buffer */
	initbuf(Qbuf, TREEMAX, TREEOFLO, &neederr);	/* reset tree allocation */
	Ingerr = 0;				/* clear error condition */
	Patflag = 0;				/* reset pattern matching flag */
	Pars = 1;				/* set scanner into "parser" mode */
	Lcase = Dcase;				/* set case mapping to default */
	Agflag = 0;				/* reset aggregate flag */
	Opflag = 0;				/* reset qmode flag */
	Resrng = 0;				/* reset result relation ptr */
	Qlflag = 0;				/* reset qualification flag */
	freesym();				/* free symbol table space */
	rngreset();				/* reset used bits in range tbl */
}

/*
** ENDQUELST -- finish command checking and processing for each quel statement
*/
endquelst(op1)
int	op1;
{
	register int			i;
	register int			op;
	char				ibuf[2];	/* two char buffer for index keys */

	op = op1;

	/* check next token for GOVAL if the next token has been read */
	switch (op)
	{
	  case mdSAVE:
	  case mdCOPY:
	  case mdCREATE:

#	  ifdef	DISTRIB
	  case mdDCREATE:
#	  endif

	  case mdINDEX:
	  case mdRANGE:
		break;

	  default:
		/* has vble ending and therefore must detect valid end of command */
		if (Lastok.tokop != GOVAL)
			/* next token not start of command */
			yyerror(NXTCMDERR, 0);
		break;
	}
	if (Agflag >= MAXAGG)
		/* too many aggregates */
		yyerror(AGGXTRA, 0);

	/* command ok so far, finish up */
	switch (op)
	{
	  case mdRETR:
	  case mdRET_UNI:
	  case mdVIEW:
		if (Resrng)
		{
			/* need to do create, everything is ready */
			call_p(mdCREATE, EXEC_DBU);
			cleanrel(&Desc);
		}
		else if (!Equel)
			/* need to print header */
			header();
		if (Ingerr)
		{
			/*
			** might be nice to back out the create already done
			** by this point so that the user doesn't need to
			*/
			endgo();	/* abort rest of go-block */
			reset();
		}
		/* fall through */

	  case mdAPP:
	  case mdDEL:
	  case mdREPL:
		if (op != mdVIEW)
		{
			call_tree(op, EXEC_DECOMP);
			break;
		}

#	  ifdef DISTRIB
	  case mdDISTRIB:
		op = mdVIEW;
#	  endif
		/* else, do VIEW */
		initp();
		setp(Resrng->relnm);
		call_tree(mdDEFINE, EXEC_QRYMOD);
		call_p(op, EXEC_QRYMOD);
		break;

	  case mdINTEG:
	  case mdPROT:
		call_tree(mdDEFINE, EXEC_QRYMOD);
		call_p(op, EXEC_QRYMOD);
		break;

	  case mdCREATE:

#	  ifdef	DISTRIB
	  case mdDCREATE:
#	  endif

	  case mdDESTROY:
	  case mdMODIFY:
		call_p(op, EXEC_DBU);
		cleanrel(&Desc);
		break;

	  case mdCOPY:
	  case mdHELP:
	  case mdPRINT:
	  case mdSAVE:
	  case mdDISPLAY:
	  case mdREMQM:
		call_p(op, EXEC_DBU);
		break;

	  case mdRANGE:
		break;

	  case mdINDEX:
		call_p(op, EXEC_DBU);
		if (Ingerr)
		{
			endgo();	/* abort rest of go-block */
			reset();
		}
		if (Indexspec)
		{
			setp(Indexname);
			setp(Indexspec);
			setp("num");
			for (i = 1; i <= Rsdmno; i++)
			{
				ibuf[0] = i & I1MASK;
				ibuf[1] = '\0';
				setp(ibuf);
			}
			call_p(mdMODIFY, EXEC_DBU);
		}
		break;
	}

	/* refresh relstat bits if necessary */
	rngfresh(op);
	init_quelst();
}

/*
** SYNCUP -- sync with process above
**
**	send tuple count if present
*/
syncup()
{
	struct pipfrmt		p;
	register struct pipfrmt	*pp;

	pp = &p;
	wrpipe(P_PRIME, pp, 0, 0, 0);
	if (Lastcnt != NULL)
	{
		wrpipe(P_NORM, pp, W_up, Lastcnt, sizeof(*Lastcnt));
	}
	wrpipe(P_END, pp, W_up);
}

/*
** SYNCDN -- wait for sync from below
**
**	save tuple count if present
*/
syncdn()
{
	struct retcode		up;
	struct pipfrmt		p;
	register struct pipfrmt	*pp;

	pp = &p;
	rdpipe(P_PRIME, pp);
	if (rdpipe(P_NORM, pp, R_down, &up, sizeof(up)) != 0)
	{
		bmove(&up, &Lastcsp, sizeof(up));
		Lastcnt = &Lastcsp;
#		ifdef	xPTR3
		if (tTf(60, 1))
			printf("Reading tuple count: %s\n", locv(Lastcnt->rc_tupcount));
#		endif
	}
}

/*
** STARTGO -- do whatever needs doing to set up a go-block
*/
startgo()
{
	/* initialize for go-block */
	getscr(1);		/* prime the scanner input */
	Lastcnt = NULL;		/* reset ptr to last tuple count */
	init_quelst();		/* most other init's are done for each statement */
	yyline = 1;		/* reset line counter */
	yypflag = 1;		/* reset action stmnts */
}

/*
** ENDGO -- do whatever needs doing to clean up after a go block
*/
endgo()
{
	getscr(2);
	syncup();
}