xref: /freebsd-src/contrib/sendmail/libsm/fopen.c (revision e2c0e292e8a7ca00ba99bcfccc9e637f45c3e8b1)
140266059SGregory Neil Shapiro /*
25dd76dd0SGregory Neil Shapiro  * Copyright (c) 2000-2002, 2004 Proofpoint, Inc. and its suppliers.
340266059SGregory Neil Shapiro  *      All rights reserved.
440266059SGregory Neil Shapiro  * Copyright (c) 1990, 1993
540266059SGregory Neil Shapiro  *	The Regents of the University of California.  All rights reserved.
640266059SGregory Neil Shapiro  *
740266059SGregory Neil Shapiro  * This code is derived from software contributed to Berkeley by
840266059SGregory Neil Shapiro  * Chris Torek.
940266059SGregory Neil Shapiro  *
1040266059SGregory Neil Shapiro  * By using this file, you agree to the terms and conditions set
1140266059SGregory Neil Shapiro  * forth in the LICENSE file which can be found at the top level of
1240266059SGregory Neil Shapiro  * the sendmail distribution.
1340266059SGregory Neil Shapiro  */
1440266059SGregory Neil Shapiro 
1540266059SGregory Neil Shapiro #include <sm/gen.h>
164313cc83SGregory Neil Shapiro SM_RCSID("@(#)$Id: fopen.c,v 1.63 2013-11-22 20:51:42 ca Exp $")
1740266059SGregory Neil Shapiro #include <errno.h>
1840266059SGregory Neil Shapiro #include <setjmp.h>
194e4196cbSGregory Neil Shapiro #include <sm/time.h>
2040266059SGregory Neil Shapiro #include <sm/heap.h>
2140266059SGregory Neil Shapiro #include <sm/signal.h>
2240266059SGregory Neil Shapiro #include <sm/assert.h>
2340266059SGregory Neil Shapiro #include <sm/io.h>
2440266059SGregory Neil Shapiro #include <sm/clock.h>
2540266059SGregory Neil Shapiro #include "local.h"
2640266059SGregory Neil Shapiro 
27b6bacd31SGregory Neil Shapiro static void	openalrm __P((int));
28b6bacd31SGregory Neil Shapiro static void	reopenalrm __P((int));
2940266059SGregory Neil Shapiro extern int      sm_io_fclose __P((SM_FILE_T *));
3040266059SGregory Neil Shapiro 
3140266059SGregory Neil Shapiro static jmp_buf OpenTimeOut, ReopenTimeOut;
3240266059SGregory Neil Shapiro 
3340266059SGregory Neil Shapiro /*
3440266059SGregory Neil Shapiro **  OPENALRM -- handler when timeout activated for sm_io_open()
3540266059SGregory Neil Shapiro **
3640266059SGregory Neil Shapiro **  Returns flow of control to where setjmp(OpenTimeOut) was set.
3740266059SGregory Neil Shapiro **
3840266059SGregory Neil Shapiro **	Parameters:
3940266059SGregory Neil Shapiro **		sig -- unused
4040266059SGregory Neil Shapiro **
4140266059SGregory Neil Shapiro **	Returns:
4240266059SGregory Neil Shapiro **		does not return
4340266059SGregory Neil Shapiro **
4440266059SGregory Neil Shapiro **	Side Effects:
4540266059SGregory Neil Shapiro **		returns flow of control to setjmp(OpenTimeOut).
4640266059SGregory Neil Shapiro **
4740266059SGregory Neil Shapiro **	NOTE:	THIS CAN BE CALLED FROM A SIGNAL HANDLER.  DO NOT ADD
4840266059SGregory Neil Shapiro **		ANYTHING TO THIS ROUTINE UNLESS YOU KNOW WHAT YOU ARE
4940266059SGregory Neil Shapiro **		DOING.
5040266059SGregory Neil Shapiro */
5140266059SGregory Neil Shapiro 
5240266059SGregory Neil Shapiro /* ARGSUSED0 */
5340266059SGregory Neil Shapiro static void
openalrm(sig)5440266059SGregory Neil Shapiro openalrm(sig)
5540266059SGregory Neil Shapiro 	int sig;
5640266059SGregory Neil Shapiro {
5740266059SGregory Neil Shapiro 	longjmp(OpenTimeOut, 1);
5840266059SGregory Neil Shapiro }
5940266059SGregory Neil Shapiro /*
6040266059SGregory Neil Shapiro **  REOPENALRM -- handler when timeout activated for sm_io_reopen()
6140266059SGregory Neil Shapiro **
6240266059SGregory Neil Shapiro **  Returns flow of control to where setjmp(ReopenTimeOut) was set.
6340266059SGregory Neil Shapiro **
6440266059SGregory Neil Shapiro **	Parameters:
6540266059SGregory Neil Shapiro **		sig -- unused
6640266059SGregory Neil Shapiro **
6740266059SGregory Neil Shapiro **	Returns:
6840266059SGregory Neil Shapiro **		does not return
6940266059SGregory Neil Shapiro **
7040266059SGregory Neil Shapiro **	Side Effects:
7140266059SGregory Neil Shapiro **		returns flow of control to setjmp(ReopenTimeOut).
7240266059SGregory Neil Shapiro **
7340266059SGregory Neil Shapiro **	NOTE:	THIS CAN BE CALLED FROM A SIGNAL HANDLER.  DO NOT ADD
7440266059SGregory Neil Shapiro **		ANYTHING TO THIS ROUTINE UNLESS YOU KNOW WHAT YOU ARE
7540266059SGregory Neil Shapiro **		DOING.
7640266059SGregory Neil Shapiro */
7740266059SGregory Neil Shapiro 
7840266059SGregory Neil Shapiro /* ARGSUSED0 */
7940266059SGregory Neil Shapiro static void
reopenalrm(sig)8040266059SGregory Neil Shapiro reopenalrm(sig)
8140266059SGregory Neil Shapiro 	int sig;
8240266059SGregory Neil Shapiro {
8340266059SGregory Neil Shapiro 	longjmp(ReopenTimeOut, 1);
8440266059SGregory Neil Shapiro }
8540266059SGregory Neil Shapiro 
8640266059SGregory Neil Shapiro /*
8740266059SGregory Neil Shapiro **  SM_IO_OPEN -- open a file of a specific type
8840266059SGregory Neil Shapiro **
8940266059SGregory Neil Shapiro **	Parameters:
9040266059SGregory Neil Shapiro **		type -- type of file to open
9140266059SGregory Neil Shapiro **		timeout -- time to complete the open
92*5b0945b5SGregory Neil Shapiro **		info -- info describing what is to be opened (type dependent)
9340266059SGregory Neil Shapiro **		flags -- user selected flags
9440266059SGregory Neil Shapiro **		rpool -- pointer to rpool to be used for this open
9540266059SGregory Neil Shapiro **
9640266059SGregory Neil Shapiro **	Returns:
9740266059SGregory Neil Shapiro **		Raises exception on heap exhaustion.
9840266059SGregory Neil Shapiro **		Aborts if type is invalid.
9940266059SGregory Neil Shapiro **		Returns NULL and sets errno
10040266059SGregory Neil Shapiro **			- when the type specific open fails
10140266059SGregory Neil Shapiro **			- when open vector errors
10240266059SGregory Neil Shapiro **			- when flags not set or invalid
10340266059SGregory Neil Shapiro **		Success returns a file pointer to the opened file type.
10440266059SGregory Neil Shapiro */
10540266059SGregory Neil Shapiro 
10640266059SGregory Neil Shapiro SM_FILE_T *
sm_io_open(type,timeout,info,flags,rpool)10740266059SGregory Neil Shapiro sm_io_open(type, timeout, info, flags, rpool)
10840266059SGregory Neil Shapiro 	const SM_FILE_T *type;
10940266059SGregory Neil Shapiro 	int SM_NONVOLATILE timeout;	/* this is not the file type timeout */
11040266059SGregory Neil Shapiro 	const void *info;
11140266059SGregory Neil Shapiro 	int flags;
11240266059SGregory Neil Shapiro 	const void *rpool;
11340266059SGregory Neil Shapiro {
11440266059SGregory Neil Shapiro 	register SM_FILE_T *fp;
11540266059SGregory Neil Shapiro 	int ioflags;
11640266059SGregory Neil Shapiro 	SM_EVENT *evt = NULL;
11740266059SGregory Neil Shapiro 
11840266059SGregory Neil Shapiro 	ioflags = sm_flags(flags);
11940266059SGregory Neil Shapiro 
12040266059SGregory Neil Shapiro 	if (ioflags == 0)
12140266059SGregory Neil Shapiro 	{
12240266059SGregory Neil Shapiro 		/* must give some indication/intent */
12340266059SGregory Neil Shapiro 		errno = EINVAL;
12440266059SGregory Neil Shapiro 		return NULL;
12540266059SGregory Neil Shapiro 	}
12640266059SGregory Neil Shapiro 
12740266059SGregory Neil Shapiro 	if (timeout == SM_TIME_DEFAULT)
12840266059SGregory Neil Shapiro 		timeout = SM_TIME_FOREVER;
12940266059SGregory Neil Shapiro 	if (timeout == SM_TIME_IMMEDIATE)
13040266059SGregory Neil Shapiro 	{
13140266059SGregory Neil Shapiro 		errno = EAGAIN;
13240266059SGregory Neil Shapiro 		return NULL;
13340266059SGregory Neil Shapiro 	}
13440266059SGregory Neil Shapiro 
13540266059SGregory Neil Shapiro 	fp = sm_fp(type, ioflags, NULL);
13640266059SGregory Neil Shapiro 
13740266059SGregory Neil Shapiro 	/*  Okay, this is where we set the timeout.  */
13840266059SGregory Neil Shapiro 	if (timeout != SM_TIME_FOREVER)
13940266059SGregory Neil Shapiro 	{
14040266059SGregory Neil Shapiro 		if (setjmp(OpenTimeOut) != 0)
14140266059SGregory Neil Shapiro 		{
14240266059SGregory Neil Shapiro 			errno = EAGAIN;
14340266059SGregory Neil Shapiro 			return NULL;
14440266059SGregory Neil Shapiro 		}
14540266059SGregory Neil Shapiro 		evt = sm_seteventm(timeout, openalrm, 0);
14640266059SGregory Neil Shapiro 	}
14740266059SGregory Neil Shapiro 
14840266059SGregory Neil Shapiro 	if ((*fp->f_open)(fp, info, flags, rpool) < 0)
14940266059SGregory Neil Shapiro 	{
15040266059SGregory Neil Shapiro 		fp->f_flags = 0;	/* release */
15140266059SGregory Neil Shapiro 		fp->sm_magic = NULL;	/* release */
15240266059SGregory Neil Shapiro 		return NULL;
15340266059SGregory Neil Shapiro 	}
15440266059SGregory Neil Shapiro 
15540266059SGregory Neil Shapiro 	/*  We're back. So undo our timeout and handler */
15640266059SGregory Neil Shapiro 	if (evt != NULL)
15740266059SGregory Neil Shapiro 		sm_clrevent(evt);
15840266059SGregory Neil Shapiro 
15940266059SGregory Neil Shapiro #if SM_RPOOL
16040266059SGregory Neil Shapiro 	if (rpool != NULL)
16140266059SGregory Neil Shapiro 		sm_rpool_attach_x(rpool, sm_io_fclose, fp);
162*5b0945b5SGregory Neil Shapiro #endif
16340266059SGregory Neil Shapiro 
16440266059SGregory Neil Shapiro 	return fp;
16540266059SGregory Neil Shapiro }
16640266059SGregory Neil Shapiro /*
16740266059SGregory Neil Shapiro **  SM_IO_DUP -- duplicate a file pointer
16840266059SGregory Neil Shapiro **
16940266059SGregory Neil Shapiro **	Parameters:
17040266059SGregory Neil Shapiro **		fp -- file pointer to duplicate
17140266059SGregory Neil Shapiro **
17240266059SGregory Neil Shapiro **	Returns:
17340266059SGregory Neil Shapiro **		Success - the duplicated file pointer
17440266059SGregory Neil Shapiro **		Failure - NULL (was an invalid file pointer or too many open)
17540266059SGregory Neil Shapiro **
17640266059SGregory Neil Shapiro **	Increments the duplicate counter (dup_cnt) for the open file pointer.
17740266059SGregory Neil Shapiro **	The counter counts the number of duplicates. When the duplicate
17840266059SGregory Neil Shapiro **	counter is 0 (zero) then the file pointer is the only one left
17940266059SGregory Neil Shapiro **	(no duplicates, it is the only one).
18040266059SGregory Neil Shapiro */
18140266059SGregory Neil Shapiro 
18240266059SGregory Neil Shapiro SM_FILE_T *
sm_io_dup(fp)18340266059SGregory Neil Shapiro sm_io_dup(fp)
18440266059SGregory Neil Shapiro 	SM_FILE_T *fp;
18540266059SGregory Neil Shapiro {
18640266059SGregory Neil Shapiro 
18740266059SGregory Neil Shapiro 	SM_REQUIRE_ISA(fp, SmFileMagic);
18840266059SGregory Neil Shapiro 	if (fp->sm_magic != SmFileMagic)
18940266059SGregory Neil Shapiro 	{
19040266059SGregory Neil Shapiro 		errno = EBADF;
19140266059SGregory Neil Shapiro 		return NULL;
19240266059SGregory Neil Shapiro 	}
19340266059SGregory Neil Shapiro 	if (fp->f_dup_cnt >= INT_MAX - 1)
19440266059SGregory Neil Shapiro 	{
19540266059SGregory Neil Shapiro 		/* Can't let f_dup_cnt wrap! */
19640266059SGregory Neil Shapiro 		errno = EMFILE;
19740266059SGregory Neil Shapiro 		return NULL;
19840266059SGregory Neil Shapiro 	}
19940266059SGregory Neil Shapiro 	fp->f_dup_cnt++;
20040266059SGregory Neil Shapiro 	return fp;
20140266059SGregory Neil Shapiro }
20240266059SGregory Neil Shapiro /*
20340266059SGregory Neil Shapiro **  SM_IO_REOPEN -- open a new file using the old file pointer
20440266059SGregory Neil Shapiro **
20540266059SGregory Neil Shapiro **	Parameters:
20640266059SGregory Neil Shapiro **		type -- file type to be opened
20740266059SGregory Neil Shapiro **		timeout -- time to complete the reopen
208*5b0945b5SGregory Neil Shapiro **		info -- information about what is to be "re-opened" (type dep.)
20940266059SGregory Neil Shapiro **		flags -- user flags to map to internal flags
21040266059SGregory Neil Shapiro **		rpool -- rpool file to be associated with
21140266059SGregory Neil Shapiro **		fp -- the file pointer to reuse
21240266059SGregory Neil Shapiro **
21340266059SGregory Neil Shapiro **	Returns:
21440266059SGregory Neil Shapiro **		Raises an exception on heap exhaustion.
21540266059SGregory Neil Shapiro **		Aborts if type is invalid.
21640266059SGregory Neil Shapiro **		Failure: returns NULL
21740266059SGregory Neil Shapiro **		Success: returns "reopened" file pointer
21840266059SGregory Neil Shapiro */
21940266059SGregory Neil Shapiro 
22040266059SGregory Neil Shapiro SM_FILE_T *
sm_io_reopen(type,timeout,info,flags,rpool,fp)22140266059SGregory Neil Shapiro sm_io_reopen(type, timeout, info, flags, rpool, fp)
22240266059SGregory Neil Shapiro 	const SM_FILE_T *type;
22340266059SGregory Neil Shapiro 	int SM_NONVOLATILE timeout;
22440266059SGregory Neil Shapiro 	const void *info;
22540266059SGregory Neil Shapiro 	int flags;
22640266059SGregory Neil Shapiro 	const void *rpool;
22740266059SGregory Neil Shapiro 	SM_FILE_T *fp;
22840266059SGregory Neil Shapiro {
22940266059SGregory Neil Shapiro 	int ioflags, ret;
23040266059SGregory Neil Shapiro 	SM_FILE_T *fp2;
23140266059SGregory Neil Shapiro 	SM_EVENT *evt = NULL;
23240266059SGregory Neil Shapiro 
23340266059SGregory Neil Shapiro 	if ((ioflags = sm_flags(flags)) == 0)
23440266059SGregory Neil Shapiro 	{
23540266059SGregory Neil Shapiro 		(void) sm_io_close(fp, timeout);
23640266059SGregory Neil Shapiro 		return NULL;
23740266059SGregory Neil Shapiro 	}
23840266059SGregory Neil Shapiro 
23940266059SGregory Neil Shapiro 	if (!Sm_IO_DidInit)
24040266059SGregory Neil Shapiro 		sm_init();
24140266059SGregory Neil Shapiro 
24240266059SGregory Neil Shapiro 	if (timeout == SM_TIME_DEFAULT)
24340266059SGregory Neil Shapiro 		timeout = SM_TIME_FOREVER;
24440266059SGregory Neil Shapiro 	if (timeout == SM_TIME_IMMEDIATE)
24540266059SGregory Neil Shapiro 	{
24640266059SGregory Neil Shapiro 		/*
24740266059SGregory Neil Shapiro 		**  Filling the buffer will take time and we are wanted to
24840266059SGregory Neil Shapiro 		**  return immediately. So...
24940266059SGregory Neil Shapiro 		*/
25040266059SGregory Neil Shapiro 
25140266059SGregory Neil Shapiro 		errno = EAGAIN;
25240266059SGregory Neil Shapiro 		return NULL;
25340266059SGregory Neil Shapiro 	}
25440266059SGregory Neil Shapiro 	/*  Okay, this is where we set the timeout.  */
25540266059SGregory Neil Shapiro 	if (timeout != SM_TIME_FOREVER)
25640266059SGregory Neil Shapiro 	{
25740266059SGregory Neil Shapiro 		if (setjmp(ReopenTimeOut) != 0)
25840266059SGregory Neil Shapiro 		{
25940266059SGregory Neil Shapiro 			errno = EAGAIN;
26040266059SGregory Neil Shapiro 			return NULL;
26140266059SGregory Neil Shapiro 		}
26240266059SGregory Neil Shapiro 
26340266059SGregory Neil Shapiro 		evt = sm_seteventm(timeout, reopenalrm, 0);
26440266059SGregory Neil Shapiro 	}
26540266059SGregory Neil Shapiro 
26640266059SGregory Neil Shapiro 	/*
26740266059SGregory Neil Shapiro 	**  There are actually programs that depend on being able to "reopen"
26840266059SGregory Neil Shapiro 	**  descriptors that weren't originally open.  Keep this from breaking.
26940266059SGregory Neil Shapiro 	**  Remember whether the stream was open to begin with, and which file
27040266059SGregory Neil Shapiro 	**  descriptor (if any) was associated with it.  If it was attached to
27140266059SGregory Neil Shapiro 	**  a descriptor, defer closing it; reopen("/dev/stdin", "r", stdin)
27240266059SGregory Neil Shapiro 	**  should work.  This is unnecessary if it was not a Unix file.
27340266059SGregory Neil Shapiro 	*/
27440266059SGregory Neil Shapiro 
27540266059SGregory Neil Shapiro 	if (fp != NULL)
27640266059SGregory Neil Shapiro 	{
27740266059SGregory Neil Shapiro 		if (fp->sm_magic != SmFileMagic)
27840266059SGregory Neil Shapiro 			fp->f_flags = SMFEOF;	/* hold on to it */
27940266059SGregory Neil Shapiro 		else
28040266059SGregory Neil Shapiro 		{
28140266059SGregory Neil Shapiro 			/* flush the stream; ANSI doesn't require this. */
28240266059SGregory Neil Shapiro 			(void) sm_io_flush(fp, SM_TIME_FOREVER);
28340266059SGregory Neil Shapiro 			(void) sm_io_close(fp, SM_TIME_FOREVER);
28440266059SGregory Neil Shapiro 		}
28540266059SGregory Neil Shapiro 	}
28640266059SGregory Neil Shapiro 
28740266059SGregory Neil Shapiro 	fp2 = sm_fp(type, ioflags, fp);
28840266059SGregory Neil Shapiro 	ret = (*fp2->f_open)(fp2, info, flags, rpool);
28940266059SGregory Neil Shapiro 
29040266059SGregory Neil Shapiro 	/*  We're back. So undo our timeout and handler */
29140266059SGregory Neil Shapiro 	if (evt != NULL)
29240266059SGregory Neil Shapiro 		sm_clrevent(evt);
29340266059SGregory Neil Shapiro 
29440266059SGregory Neil Shapiro 	if (ret < 0)
29540266059SGregory Neil Shapiro 	{
29640266059SGregory Neil Shapiro 		fp2->f_flags = 0;	/* release */
29740266059SGregory Neil Shapiro 		fp2->sm_magic = NULL;	/* release */
29840266059SGregory Neil Shapiro 		return NULL;
29940266059SGregory Neil Shapiro 	}
30040266059SGregory Neil Shapiro 
30140266059SGregory Neil Shapiro 	/*
30240266059SGregory Neil Shapiro 	**  We're not preserving this logic (below) for sm_io because it is now
30340266059SGregory Neil Shapiro 	**  abstracted at least one "layer" away. By closing and reopening
30440266059SGregory Neil Shapiro 	**  the 1st fd used should be the just released one (when Unix
30540266059SGregory Neil Shapiro 	**  behavior followed). Old comment::
30640266059SGregory Neil Shapiro 	**  If reopening something that was open before on a real file, try
30740266059SGregory Neil Shapiro 	**  to maintain the descriptor.  Various C library routines (perror)
30840266059SGregory Neil Shapiro 	**  assume stderr is always fd STDERR_FILENO, even if being reopen'd.
30940266059SGregory Neil Shapiro 	*/
31040266059SGregory Neil Shapiro 
31140266059SGregory Neil Shapiro #if SM_RPOOL
31240266059SGregory Neil Shapiro 	if (rpool != NULL)
31340266059SGregory Neil Shapiro 		sm_rpool_attach_x(rpool, sm_io_close, fp2);
314*5b0945b5SGregory Neil Shapiro #endif
31540266059SGregory Neil Shapiro 
31640266059SGregory Neil Shapiro 	return fp2;
31740266059SGregory Neil Shapiro }
31840266059SGregory Neil Shapiro /*
31940266059SGregory Neil Shapiro **  SM_IO_AUTOFLUSH -- link another file to this for auto-flushing
32040266059SGregory Neil Shapiro **
32140266059SGregory Neil Shapiro **	When a read occurs on fp, fp2 will be flushed iff there is no
32240266059SGregory Neil Shapiro **	data waiting on fp.
32340266059SGregory Neil Shapiro **
32440266059SGregory Neil Shapiro **	Parameters:
32540266059SGregory Neil Shapiro **		fp -- the file opened for reading.
32640266059SGregory Neil Shapiro **		fp2 -- the file opened for writing.
32740266059SGregory Neil Shapiro **
32840266059SGregory Neil Shapiro **	Returns:
32940266059SGregory Neil Shapiro **		The old flush file pointer.
33040266059SGregory Neil Shapiro */
33140266059SGregory Neil Shapiro 
33240266059SGregory Neil Shapiro SM_FILE_T *
sm_io_autoflush(fp,fp2)33340266059SGregory Neil Shapiro sm_io_autoflush(fp, fp2)
33440266059SGregory Neil Shapiro 	SM_FILE_T *fp;
33540266059SGregory Neil Shapiro 	SM_FILE_T *fp2;
33640266059SGregory Neil Shapiro {
33740266059SGregory Neil Shapiro 	SM_FILE_T *savefp;
33840266059SGregory Neil Shapiro 
33940266059SGregory Neil Shapiro 	SM_REQUIRE_ISA(fp, SmFileMagic);
34040266059SGregory Neil Shapiro 	if (fp2 != NULL)
34140266059SGregory Neil Shapiro 		SM_REQUIRE_ISA(fp2, SmFileMagic);
34240266059SGregory Neil Shapiro 
34340266059SGregory Neil Shapiro 	savefp = fp->f_flushfp;
34440266059SGregory Neil Shapiro 	fp->f_flushfp = fp2;
34540266059SGregory Neil Shapiro 	return savefp;
34640266059SGregory Neil Shapiro }
34740266059SGregory Neil Shapiro /*
34840266059SGregory Neil Shapiro **  SM_IO_AUTOMODE -- link another file to this for auto-moding
34940266059SGregory Neil Shapiro **
35040266059SGregory Neil Shapiro **	When the mode (blocking or non-blocking) changes for fp1 then
35140266059SGregory Neil Shapiro **	update fp2's mode at the same time. This is to be used when
35240266059SGregory Neil Shapiro **	a system dup() has generated a second file descriptor for
35340266059SGregory Neil Shapiro **	another sm_io_open() by file descriptor. The modes have been
35440266059SGregory Neil Shapiro **	linked in the system and this formalizes it for sm_io internally.
35540266059SGregory Neil Shapiro **
35640266059SGregory Neil Shapiro **	Parameters:
35740266059SGregory Neil Shapiro **		fp1 -- the first file
35840266059SGregory Neil Shapiro **		fp2 -- the second file
35940266059SGregory Neil Shapiro **
36040266059SGregory Neil Shapiro **	Returns:
36140266059SGregory Neil Shapiro **		nothing
36240266059SGregory Neil Shapiro */
36340266059SGregory Neil Shapiro 
36440266059SGregory Neil Shapiro void
sm_io_automode(fp1,fp2)36540266059SGregory Neil Shapiro sm_io_automode(fp1, fp2)
36640266059SGregory Neil Shapiro 	SM_FILE_T *fp1;
36740266059SGregory Neil Shapiro 	SM_FILE_T *fp2;
36840266059SGregory Neil Shapiro {
36940266059SGregory Neil Shapiro 	SM_REQUIRE_ISA(fp1, SmFileMagic);
37040266059SGregory Neil Shapiro 	SM_REQUIRE_ISA(fp2, SmFileMagic);
37140266059SGregory Neil Shapiro 
37240266059SGregory Neil Shapiro 	fp1->f_modefp = fp2;
37340266059SGregory Neil Shapiro 	fp2->f_modefp = fp1;
37440266059SGregory Neil Shapiro }
375