Blame src/lib/libast/sfio/sfstack.c

Packit Service a8c26c
/***********************************************************************
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
*               This software is part of the ast package               *
Packit Service a8c26c
*          Copyright (c) 1985-2011 AT&T Intellectual Property          *
Packit Service a8c26c
*                      and is licensed under the                       *
Packit Service a8c26c
*                 Eclipse Public License, Version 1.0                  *
Packit Service a8c26c
*                    by AT&T Intellectual Property                     *
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
*                A copy of the License is available at                 *
Packit Service a8c26c
*          http://www.eclipse.org/org/documents/epl-v10.html           *
Packit Service a8c26c
*         (with md5 checksum b35adb5213ca9657e911e9befb180842)         *
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
*              Information and Software Systems Research               *
Packit Service a8c26c
*                            AT&T Research                             *
Packit Service a8c26c
*                           Florham Park NJ                            *
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
*                 Glenn Fowler <gsf@research.att.com>                  *
Packit Service a8c26c
*                  David Korn <dgk@research.att.com>                   *
Packit Service a8c26c
*                   Phong Vo <kpv@research.att.com>                    *
Packit Service a8c26c
*                                                                      *
Packit Service a8c26c
***********************************************************************/
Packit Service a8c26c
#include	"sfhdr.h"
Packit Service a8c26c
Packit Service a8c26c
Packit Service a8c26c
/*	Push/pop streams
Packit Service a8c26c
**
Packit Service a8c26c
**	Written by Kiem-Phong Vo.
Packit Service a8c26c
*/
Packit Service a8c26c
Packit Service a8c26c
#define STKMTXLOCK(f1,f2) \
Packit Service a8c26c
	{ if(f1) SFMTXLOCK(f1); \
Packit Service a8c26c
	  if(f2) SFMTXLOCK(f2); \
Packit Service a8c26c
	}
Packit Service a8c26c
#define STKMTXRETURN(f1,f2,rv) \
Packit Service a8c26c
	{ if(f1) SFMTXUNLOCK(f1); \
Packit Service a8c26c
	  if(f2) SFMTXUNLOCK(f2); \
Packit Service a8c26c
	  return(rv); \
Packit Service a8c26c
	}
Packit Service a8c26c
Packit Service a8c26c
#if __STD_C
Packit Service a8c26c
Sfio_t* sfstack(Sfio_t* f1, Sfio_t* f2)
Packit Service a8c26c
#else
Packit Service a8c26c
Sfio_t* sfstack(f1,f2)
Packit Service a8c26c
Sfio_t*	f1;	/* base of stack	*/
Packit Service a8c26c
Sfio_t*	f2;	/* top of stack	*/
Packit Service a8c26c
#endif
Packit Service a8c26c
{
Packit Service a8c26c
	reg int		n;
Packit Service a8c26c
	reg Sfio_t*	rf;
Packit Service a8c26c
	reg Sfrsrv_t*	rsrv;
Packit Service a8c26c
	reg Void_t*	mtx;
Packit Service a8c26c
Packit Service a8c26c
	STKMTXLOCK(f1,f2);
Packit Service a8c26c
Packit Service a8c26c
	if(f1 && (f1->mode&SF_RDWR) != f1->mode && _sfmode(f1,0,0) < 0)
Packit Service a8c26c
		STKMTXRETURN(f1,f2, NIL(Sfio_t*));
Packit Service a8c26c
	if(f2 && (f2->mode&SF_RDWR) != f2->mode && _sfmode(f2,0,0) < 0)
Packit Service a8c26c
		STKMTXRETURN(f1,f2, NIL(Sfio_t*));
Packit Service a8c26c
	if(!f1)
Packit Service a8c26c
		STKMTXRETURN(f1,f2, f2);
Packit Service a8c26c
Packit Service a8c26c
	/* give access to other internal functions */
Packit Service a8c26c
	_Sfstack = sfstack;
Packit Service a8c26c
Packit Service a8c26c
	if(f2 == SF_POPSTACK)
Packit Service a8c26c
	{	if(!(f2 = f1->push))
Packit Service a8c26c
			STKMTXRETURN(f1,f2, NIL(Sfio_t*));
Packit Service a8c26c
		f2->mode &= ~SF_PUSH;
Packit Service a8c26c
	}
Packit Service a8c26c
	else
Packit Service a8c26c
	{	if(f2->push)
Packit Service a8c26c
			STKMTXRETURN(f1,f2, NIL(Sfio_t*));
Packit Service a8c26c
		if(f1->pool && f1->pool != &_Sfpool && f1->pool != f2->pool &&
Packit Service a8c26c
		   f1 == f1->pool->sf[0])
Packit Service a8c26c
		{	/* get something else to pool front since f1 will be locked */
Packit Service a8c26c
			for(n = 1; n < f1->pool->n_sf; ++n)
Packit Service a8c26c
			{	if(SFFROZEN(f1->pool->sf[n]) )
Packit Service a8c26c
					continue;
Packit Service a8c26c
				(*_Sfpmove)(f1->pool->sf[n],0);
Packit Service a8c26c
				break;
Packit Service a8c26c
			}
Packit Service a8c26c
		}
Packit Service a8c26c
	}
Packit Service a8c26c
Packit Service a8c26c
	if(f2->pool && f2->pool != &_Sfpool && f2 != f2->pool->sf[0])
Packit Service a8c26c
		(*_Sfpmove)(f2,0);
Packit Service a8c26c
Packit Service a8c26c
	/* swap streams */
Packit Service a8c26c
	sfswap(f1,f2);
Packit Service a8c26c
Packit Service a8c26c
	/* but the reserved buffer and mutex must remain the same */
Packit Service a8c26c
	rsrv = f1->rsrv; f1->rsrv = f2->rsrv; f2->rsrv = rsrv;
Packit Service a8c26c
	mtx = f1->mutex; f1->mutex = f2->mutex; f2->mutex = mtx;
Packit Service a8c26c
Packit Service a8c26c
	SFLOCK(f1,0);
Packit Service a8c26c
	SFLOCK(f2,0);
Packit Service a8c26c
Packit Service a8c26c
	if(f2->push != f2)
Packit Service a8c26c
	{	/* freeze the pushed stream */
Packit Service a8c26c
		f2->mode |= SF_PUSH;
Packit Service a8c26c
		f1->push = f2;
Packit Service a8c26c
		rf = f1;
Packit Service a8c26c
	}
Packit Service a8c26c
	else
Packit Service a8c26c
	{	/* unfreeze the just exposed stream */
Packit Service a8c26c
		f1->mode &= ~SF_PUSH;
Packit Service a8c26c
		f2->push = NIL(Sfio_t*);
Packit Service a8c26c
		rf = f2;
Packit Service a8c26c
	}
Packit Service a8c26c
Packit Service a8c26c
	SFOPEN(f1,0);
Packit Service a8c26c
	SFOPEN(f2,0);
Packit Service a8c26c
Packit Service a8c26c
	STKMTXRETURN(f1,f2, rf);
Packit Service a8c26c
}