2 * Copyright (c) 2000-2001, 2004 Sendmail, Inc. and its suppliers.
4 * Copyright (c) 1990, 1993
5 * The Regents of the University of California. All rights reserved.
7 * This code is derived from software contributed to Berkeley by
10 * By using this file, you agree to the terms and conditions set
11 * forth in the LICENSE file which can be found at the top level of
12 * the sendmail distribution.
16 SM_RCSID("@(#)$Id: fpos.c,v 1.39 2005/06/14 23:07:20 ca Exp $")
21 #include <sm/signal.h>
24 #include <sm/assert.h>
27 static void tellalrm
__P((int));
28 static jmp_buf TellTimeOut
;
31 ** TELLALRM -- handler when timeout activated for sm_io_tell()
33 ** Returns flow of control to where setjmp(TellTimeOut) was set.
42 ** returns flow of control to setjmp(TellTimeOut).
44 ** NOTE: THIS CAN BE CALLED FROM A SIGNAL HANDLER. DO NOT ADD
45 ** ANYTHING TO THIS ROUTINE UNLESS YOU KNOW WHAT YOU ARE
54 longjmp(TellTimeOut
, 1);
58 ** SM_IO_TELL -- position the file pointer
61 ** fp -- the file pointer to get repositioned
62 ** timeout -- time to complete the tell (milliseconds)
65 ** Success -- the repositioned location.
66 ** Failure -- -1 (minus 1) and sets errno
70 sm_io_tell(fp
, timeout
)
71 register SM_FILE_T
*fp
;
72 int SM_NONVOLATILE timeout
;
77 SM_REQUIRE_ISA(fp
, SmFileMagic
);
78 if (fp
->f_seek
== NULL
)
80 errno
= ESPIPE
; /* historic practice */
84 if (timeout
== SM_TIME_DEFAULT
)
85 timeout
= fp
->f_timeout
;
86 if (timeout
== SM_TIME_IMMEDIATE
)
89 ** Filling the buffer will take time and we are wanted to
90 ** return immediately. So...
98 ** Find offset of underlying I/O object, then adjust byte position
99 ** may adjust seek offset on append stream
102 (void) sm_flush(fp
, (int *) &timeout
);
104 /* This is where we start the timeout */
105 if (timeout
!= SM_TIME_FOREVER
)
107 if (setjmp(TellTimeOut
) != 0)
113 evt
= sm_seteventm(timeout
, tellalrm
, 0);
116 if (fp
->f_flags
& SMOFF
)
117 pos
= fp
->f_lseekoff
;
120 /* XXX only set the timeout here? */
121 pos
= (*fp
->f_seek
)(fp
, (off_t
) 0, SM_IO_SEEK_CUR
);
125 if (fp
->f_flags
& SMRD
)
128 ** Reading. Any unread characters (including
129 ** those from ungetc) cause the position to be
130 ** smaller than that in the underlying object.
137 else if (fp
->f_flags
& SMWR
&& fp
->f_p
!= NULL
)
140 ** Writing. Any buffered characters cause the
141 ** position to be greater than that in the
142 ** underlying object.
145 pos
+= fp
->f_p
- fp
->f_bf
.smb_base
;
149 /* We're back. So undo our timeout and handler */