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