Annotation of coherent/b/lib/libc/XSTDIO/setvbuf.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * libc/stdio/setvbuf.c
        !             3:  * ANSI-compliant C standard i/o library.
        !             4:  * setvbuf()
        !             5:  * ANSI 4.9.5.6.
        !             6:  * Set i/o buffer.
        !             7:  */
        !             8: 
        !             9: #include <stdio.h>
        !            10: #include <stdlib.h>
        !            11: 
        !            12: /*
        !            13:  * The get and put function pointers in a FILE
        !            14:  * initially contain &_fginit() and &_fpinit().
        !            15:  * The buffering characteristics of the stream
        !            16:  * are determined either by an explicit setvbuf()
        !            17:  * before the first get or put on the stream
        !            18:  * or by a setvbuf() call from finit() on the first get or put.
        !            19:  * setvbuf() allocates a buffer if necessary and
        !            20:  * initializes various fields in the FILE.
        !            21:  *
        !            22:  * For unbuffered streams, _cc == 0 and _bp == _cp == _dp == _ep == NULL.
        !            23:  * For buffered streams:
        !            24:  *     _cc == 0        nothing is buffered
        !            25:  *     _cc > 0         _cc input or output characters are buffered
        !            26:  *     _bp             base of buffer
        !            27:  *     _cp             current buffer pointer
        !            28:  *     _dp             data pointer
        !            29:  *     _ep             end of buffer pointer
        !            30:  * The old version of the stdio functions used
        !            31:  *     _cc < 0         -(_cc) input characters are buffered
        !            32:  *     _cc > 0         _cc output characters are buffered
        !            33:  * which makes much more sense but is unfortunately not compatible with
        !            34:  * the Unix version of stdio.h.  This stdio gets very confused if you
        !            35:  * mix input and output calls without fflush() in between on a rw stream;
        !            36:  * this is non-ANSI compliant but works right with the old version.
        !            37:  */
        !            38: int
        !            39: setvbuf(stream, buf, mode, size) register FILE *stream; char *buf; int mode; size_t size;
        !            40: {
        !            41:        register _FILE2 *f2p;
        !            42: 
        !            43: #if    _ASCII
        !            44:        register int isascii;
        !            45: 
        !            46:        isascii = stream->_ff2 & _FASCII;
        !            47: #endif
        !            48: 
        !            49:        if (stream->_mode != _MODE_UNINIT)
        !            50:                return 1;                       /* Mode assigned, failure */
        !            51: 
        !            52:        /* Allocate a buffer if necessary. */
        !            53:        if (mode != _IONBF && buf == NULL) {
        !            54:                if ((buf = malloc(size)) == NULL)
        !            55:                        return 1;               /* Buffer allocation failure */
        !            56:                stream->_ff2 |= _FFREEB;        /* Free buffer when closed */
        !            57:        }
        !            58: 
        !            59:        f2p = stream->_f2p;
        !            60:        switch(mode) {
        !            61: 
        !            62:        case _IOFBF:                            /* Fully buffered. */
        !            63:                stream->_ff1 |= _IOFBF;
        !            64:                stream->_mode = _MODE_FBUF;
        !            65: #if    _ASCII
        !            66:                f2p->_pt = isascii ? &_fputba : &_fputb;
        !            67: #else
        !            68:                f2p->_pt = &_fputb;
        !            69: #endif
        !            70:                f2p->_dp = stream->_cp = buf + boffset(stream, size);
        !            71:        lab:
        !            72: #if    _ASCII
        !            73:                f2p->_gt = isascii ? &_fgetba : &_fgetb;
        !            74: #else
        !            75:                f2p->_gt = &_fgetb;
        !            76: #endif
        !            77:                f2p->_bp = buf;
        !            78:                f2p->_ep = buf + size;
        !            79:                break;
        !            80: 
        !            81:        case _IOLBF:                            /* Line buffered */
        !            82:                stream->_ff1 |= _IOLBF;
        !            83:                stream->_mode = _MODE_LBUF;
        !            84: #if    _ASCII
        !            85:                f2p->_pt = isascii ? &_fputta : &_fputt;
        !            86: #else
        !            87:                f2p->_pt = &_fputt;
        !            88: #endif
        !            89:                f2p->_dp = stream->_cp = buf;
        !            90:                goto lab;
        !            91: 
        !            92:        case _IONBF:                            /* Unbuffered */
        !            93:                stream->_ff1 |= _IONBF;
        !            94:                stream->_mode = _MODE_NBUF;
        !            95: #if    _ASCII
        !            96:                f2p->_gt = isascii ? &_fgetca : &_fgetc;
        !            97:                f2p->_pt = isascii ? &_fputca : &_fputc;
        !            98: #else
        !            99:                f2p->_gt = &_fgetc;
        !           100:                f2p->_pt = &_fputc;
        !           101: #endif
        !           102:                break;
        !           103: 
        !           104:        default:                                /* Unrecognized mode */
        !           105:                return 1;
        !           106:        }
        !           107: 
        !           108:        if (stream->_ff1 & _FWONLY)
        !           109:                f2p->_gt = &_fgete;
        !           110:        if (stream->_ff1 & _FRONLY)
        !           111:                f2p->_pt = &_fpute;
        !           112:        stream->_cc = 0;
        !           113:        return 0;                               /* Success */
        !           114: }
        !           115: 
        !           116: /*
        !           117:  * Return the current offset of a stream.
        !           118:  * This is an attempt to keep buffered file i/o properly aligned.
        !           119:  * For example, a 200 byte file opened in append mode with a
        !           120:  * default buffer size of 512 bytes will have boffset 200
        !           121:  * and subsequent write()s will hopefully occur at 512-byte boundaries.
        !           122:  * If the user calls setvbuf() explicitly with a strange buffer size
        !           123:  * (not a multiple of the size for optimum disk i/o),
        !           124:  * the user will get degraded performance.
        !           125:  */
        !           126: static
        !           127: int
        !           128: boffset(fp, size) register FILE *fp; register size_t size;
        !           129: {
        !           130:        register long offset;
        !           131: 
        !           132:        if ((offset=lseek(fileno(fp), 0L, SEEK_CUR)) == -1L || size == 0)
        !           133:                return 0;
        !           134:        return (unsigned)offset%(unsigned)size;
        !           135: }
        !           136: 
        !           137: /* end of libc/stdio/setvbuf.c */

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.