1 /***********************************************************************
3 * This software is part of the ast package *
4 * Copyright (c) 1985-2010 AT&T Intellectual Property *
5 * and is licensed under the *
6 * Common Public License, Version 1.0 *
7 * by AT&T Intellectual Property *
9 * A copy of the License is available at *
10 * http://www.opensource.org/licenses/cpl1.0.txt *
11 * (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) *
13 * Information and Software Systems Research *
17 * Glenn Fowler <gsf@research.att.com> *
18 * David Korn <dgk@research.att.com> *
19 * Phong Vo <kpv@research.att.com> *
21 ***********************************************************************/
23 static char* Version
= "\n@(#)$Id: sfio (AT&T Labs - Research) 2009-09-15 $\0\n";
25 /* Functions to set a given stream to some desired mode
27 ** Written by Kiem-Phong Vo.
30 ** 06/27/1990 (first version)
39 ** 08/01/1998 (extended formatting)
40 ** 09/09/1999 (thread-safe)
41 ** 02/01/2001 (adaptive buffering)
42 ** 05/31/2002 (multi-byte handling in sfvprintf/vscanf)
43 ** 09/06/2002 (SF_IOINTR flag)
44 ** 11/15/2002 (%#c for sfvprintf)
45 ** 05/31/2003 (sfsetbuf(f,f,align_size) to set alignment for data)
46 ** (%I1d is fixed to handle "signed char" correctly)
47 ** 01/01/2004 Porting issues to various platforms resolved.
48 ** 06/01/2008 Allowing notify() at entering/exiting thread-safe routines.
49 ** 09/15/2008 Add sfwalk().
52 /* the below is for protecting the application from SIGPIPE */
56 #define Sfsignal_f Sig_handler_t
59 typedef void(* Sfsignal_f
)_ARG_((int));
61 static int _Sfsigp
= 0; /* # of streams needing SIGPIPE protection */
63 /* done at exiting time */
65 static void _sfcleanup(void)
67 static void _sfcleanup()
75 f
= (Sfio_t
*)Version
; /* shut compiler warning */
77 /* set this so that no more buffering is allowed for write streams */
82 for(p
= &_Sfpool
; p
; p
= p
->next
)
83 { for(n
= 0; n
< p
->n_sf
; ++n
)
84 { if(!(f
= p
->sf
[n
]) || SFFROZEN(f
) )
90 /* let application know that we are leaving */
91 (void)SFRAISE(f
, SF_ATEXIT
, NIL(Void_t
*));
93 if(f
->flags
&SF_STRING
)
96 /* from now on, write streams are unbuffered */
97 pool
= f
->mode
&SF_POOL
;
99 if((f
->flags
&SF_WRITE
) && !(f
->mode
&SF_WRITE
))
100 (void)_sfmode(f
,SF_WRITE
,1);
101 if(((f
->bits
&SF_MMAP
) && f
->data
) ||
102 ((f
->mode
&SF_WRITE
) && f
->next
== f
->data
) )
103 (void)SFSETBUF(f
,NIL(Void_t
*),0);
112 /* put into discrete pool */
114 int _sfsetpool(Sfio_t
* f
)
125 { _Sfcleanup
= _sfcleanup
;
126 (void)atexit(_sfcleanup
);
130 p
= f
->pool
= &_Sfpool
;
136 if(p
->n_sf
>= p
->s_sf
)
137 { if(p
->s_sf
== 0) /* initialize pool array */
138 { p
->s_sf
= sizeof(p
->array
)/sizeof(p
->array
[0]);
141 else /* allocate a larger array */
142 { n
= (p
->sf
!= p
->array
? p
->s_sf
: (p
->s_sf
/4 + 1)*4) + 4;
143 if(!(array
= (Sfio_t
**)malloc(n
*sizeof(Sfio_t
*))) )
146 /* move old array to new one */
147 memcpy((Void_t
*)array
,(Void_t
*)p
->sf
,p
->n_sf
*sizeof(Sfio_t
*));
148 if(p
->sf
!= p
->array
)
149 free((Void_t
*)p
->sf
);
156 /* always add at end of array because if this was done during some sort
157 of walk thru all streams, we'll want the new stream to be seen.
159 p
->sf
[p
->n_sf
++] = f
;
163 POOLMTXRETURN(p
, rv
);
166 /* create an auxiliary buffer for sfgetr/sfreserve/sfputr */
168 Sfrsrv_t
* _sfrsrv(reg Sfio_t
* f
, reg ssize_t size
)
170 Sfrsrv_t
* _sfrsrv(f
,size
)
177 /* make buffer if nothing yet */
178 size
= ((size
+ SF_GRAIN
-1)/SF_GRAIN
)*SF_GRAIN
;
179 if(!(rsrv
= f
->rsrv
) || size
> rsrv
->size
)
180 { if(!(rs
= (Sfrsrv_t
*)malloc(size
+sizeof(Sfrsrv_t
))))
185 memcpy(rs
,rsrv
,sizeof(Sfrsrv_t
)+rsrv
->slen
);
197 return size
>= 0 ? rsrv
: NIL(Sfrsrv_t
*);
202 static void ignoresig(int sig
)
204 static void ignoresig(sig
)
208 signal(sig
, ignoresig
);
213 int _sfpopen(reg Sfio_t
* f
, int fd
, int pid
, int stdio
)
215 int _sfpopen(f
, fd
, pid
, stdio
)
219 int stdio
; /* stdio popen() does not reset SIGPIPE handler */
227 if(!(p
= f
->proc
= (Sfproc_t
*)malloc(sizeof(Sfproc_t
))) )
231 p
->size
= p
->ndata
= 0;
232 p
->rdata
= NIL(uchar
*);
234 p
->sigp
= (!stdio
&& pid
>= 0 && (f
->flags
&SF_WRITE
)) ? 1 : 0;
236 #ifdef SIGPIPE /* protect from broken pipe signal */
238 { Sfsignal_f handler
;
240 (void)vtmtxlock(_Sfmutex
);
241 if((handler
= signal(SIGPIPE
, ignoresig
)) != SIG_DFL
&&
242 handler
!= ignoresig
)
243 signal(SIGPIPE
, handler
); /* honor user handler */
245 (void)vtmtxunlock(_Sfmutex
);
253 int _sfpclose(reg Sfio_t
* f
)
256 reg Sfio_t
* f
; /* stream to close */
264 f
->proc
= NIL(Sfproc_t
*);
271 { /* close the associated stream */
275 /* wait for process termination */
277 sigcritical(SIG_REG_EXEC
|SIG_REG_PROC
);
279 while ((pid
= waitpid(p
->pid
,&status
,0)) == -1 && errno
== EINTR
)
288 (void)vtmtxlock(_Sfmutex
);
289 if(p
->sigp
&& (_Sfsigp
-= 1) <= 0)
290 { Sfsignal_f handler
;
291 if((handler
= signal(SIGPIPE
,SIG_DFL
)) != SIG_DFL
&&
292 handler
!= ignoresig
)
293 signal(SIGPIPE
,handler
); /* honor user handler */
296 (void)vtmtxunlock(_Sfmutex
);
305 static int _sfpmode(Sfio_t
* f
, int type
)
307 static int _sfpmode(f
,type
)
318 { /* save unread data */
319 p
->ndata
= f
->endb
-f
->next
;
320 if(p
->ndata
> p
->size
)
322 free((char*)p
->rdata
);
323 if((p
->rdata
= (uchar
*)malloc(p
->ndata
)) )
331 memcpy((Void_t
*)p
->rdata
,(Void_t
*)f
->next
,p
->ndata
);
335 { /* restore read data */
336 if(p
->ndata
> f
->size
) /* may lose data!!! */
339 { memcpy((Void_t
*)f
->data
,(Void_t
*)p
->rdata
,p
->ndata
);
340 f
->endb
= f
->data
+p
->ndata
;
345 /* switch file descriptor */
356 int _sfmode(reg Sfio_t
* f
, reg
int wanted
, reg
int local
)
358 int _sfmode(f
, wanted
, local
)
359 reg Sfio_t
* f
; /* change r/w mode and sync file pointer for this stream */
360 reg
int wanted
; /* desired mode */
361 reg
int local
; /* a local call */
368 SFONCE(); /* initialize mutexes */
370 if(wanted
&SF_SYNCED
) /* for (SF_SYNCED|SF_READ) stream, just junk data */
371 { wanted
&= ~SF_SYNCED
;
372 if((f
->mode
&(SF_SYNCED
|SF_READ
)) == (SF_SYNCED
|SF_READ
) )
373 { f
->next
= f
->endb
= f
->endr
= f
->data
;
374 f
->mode
&= ~SF_SYNCED
;
378 if((!local
&& SFFROZEN(f
)) || (!(f
->flags
&SF_STRING
) && f
->file
< 0))
379 { if(local
|| !f
->disc
|| !f
->disc
->exceptf
)
385 { if((rv
= (*f
->disc
->exceptf
)(f
,SF_LOCKED
,0,f
->disc
)) < 0)
387 if((!local
&& SFFROZEN(f
)) ||
388 (!(f
->flags
&SF_STRING
) && f
->file
< 0) )
400 { f
->mode
&= ~SF_GETR
;
402 if((f
->bits
&SF_MMAP
) && (f
->tiny
[0] += 1) >= (4*SF_NMAP
) )
403 { /* turn off mmap to avoid page faulting */
404 sfsetbuf(f
,(Void_t
*)f
->tiny
,(size_t)SF_UNBOUND
);
410 { f
->next
[-1] = f
->getr
;
415 if(f
->mode
&SF_STDIO
) /* synchronizing with stdio pointers */
418 if(f
->disc
== _Sfudisc
&& wanted
== SF_WRITE
&&
419 sfclose((*_Sfstack
)(f
,NIL(Sfio_t
*))) < 0 )
425 { /* move to head of pool */
426 if(f
== f
->pool
->sf
[0] || (*_Sfpmove
)(f
,0) < 0 )
435 /* buffer initialization */
439 if(!f
->pool
&& _sfsetpool(f
) < 0)
447 if(wanted
!= (int)(f
->mode
&SF_RDWR
) && !(f
->flags
&wanted
) )
450 if((f
->flags
&SF_STRING
) && f
->size
>= 0 && f
->data
)
451 { f
->mode
&= ~SF_INIT
;
452 f
->extent
= ((f
->flags
&SF_READ
) || (f
->bits
&SF_BOTH
)) ?
455 f
->endb
= f
->data
+ f
->size
;
456 f
->next
= f
->endr
= f
->endw
= f
->data
;
459 else f
->endw
= f
->endb
;
463 (void)SFSETBUF(f
,f
->data
,f
->size
);
464 f
->flags
|= (n
&SF_MALLOC
);
468 if(wanted
== (int)SFMODE(f
,1))
473 case SF_WRITE
: /* switching to SF_READ */
474 if(wanted
== 0 || wanted
== SF_WRITE
)
476 if(!(f
->flags
&SF_READ
) )
478 else if(f
->flags
&SF_STRING
)
480 f
->endb
= f
->data
+f
->extent
;
486 if(f
->next
> f
->data
&& SFFLSBUF(f
,-1) < 0)
492 f
->size
= sizeof(f
->tiny
);
494 f
->next
= f
->endr
= f
->endw
= f
->endb
= f
->data
;
495 f
->mode
= SF_READ
|SF_LOCK
;
497 /* restore saved read data for coprocess */
498 if(f
->proc
&& _sfpmode(f
,wanted
) < 0)
503 case (SF_READ
|SF_SYNCED
): /* a previously sync-ed read stream */
504 if(wanted
!= SF_WRITE
)
505 { /* just reset the pointers */
506 f
->mode
= SF_READ
|SF_LOCK
;
508 /* see if must go with new physical location */
509 if((f
->flags
&(SF_SHARE
|SF_PUBLIC
)) == (SF_SHARE
|SF_PUBLIC
) &&
510 (addr
= SFSK(f
,0,SEEK_CUR
,f
->disc
)) != f
->here
)
513 if((f
->bits
&SF_MMAP
) && f
->data
)
514 { SFMUNMAP(f
,f
->data
,f
->endb
-f
->data
);
515 f
->data
= NIL(uchar
*);
518 f
->endb
= f
->endr
= f
->endw
= f
->next
= f
->data
;
522 { addr
= f
->here
+ (f
->endb
- f
->next
);
523 if(SFSK(f
,addr
,SEEK_SET
,f
->disc
) < 0)
532 case SF_READ
: /* switching to SF_WRITE */
533 if(wanted
!= SF_WRITE
)
535 else if(!(f
->flags
&SF_WRITE
))
537 else if(f
->flags
&SF_STRING
)
538 { f
->endb
= f
->data
+f
->size
;
539 f
->mode
= SF_WRITE
|SF_LOCK
;
543 /* save unread data before switching mode */
544 if(f
->proc
&& _sfpmode(f
,wanted
) < 0)
547 /* reset buffer and seek pointer */
548 if(!(f
->mode
&SF_SYNCED
) )
549 { n
= f
->endb
- f
->next
;
550 if(f
->extent
>= 0 && (n
> 0 || (f
->data
&& (f
->bits
&SF_MMAP
))) )
551 { /* reset file pointer */
553 if(SFSK(f
,addr
,SEEK_SET
,f
->disc
) < 0)
559 f
->mode
= SF_WRITE
|SF_LOCK
;
563 SFMUNMAP(f
,f
->data
,f
->endb
-f
->data
);
564 (void)SFSETBUF(f
,(Void_t
*)f
->tiny
,(size_t)SF_UNBOUND
);
567 if(f
->data
== f
->tiny
)
568 { f
->endb
= f
->data
= f
->next
= NIL(uchar
*);
571 else f
->endb
= (f
->next
= f
->data
) + f
->size
;
575 default: /* unknown case */
577 if((wanted
&= SF_RDWR
) == 0 && (wanted
= f
->flags
&SF_RDWR
) == SF_RDWR
)
580 /* set errno for operations that access wrong stream type */
581 if(wanted
!= (f
->mode
&SF_RDWR
) && f
->file
>= 0)
584 if(_Sfnotify
) /* notify application of the error */
585 (*_Sfnotify
)(f
, wanted
, (void*)((long)f
->file
));