/** @file | |
Implementation of setvbuf as declared in <stdio.h>. | |
Copyright (c) 2010 - 2011, Intel Corporation. All rights reserved.<BR> | |
This program and the accompanying materials are licensed and made available | |
under the terms and conditions of the BSD License that accompanies this | |
distribution. The full text of the license may be found at | |
http://opensource.org/licenses/bsd-license. | |
THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS, | |
WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED. | |
Copyright (c) 1990, 1993 | |
The Regents of the University of California. All rights reserved. | |
This code is derived from software contributed to Berkeley by | |
Chris Torek. | |
Redistribution and use in source and binary forms, with or without | |
modification, are permitted provided that the following conditions | |
are met: | |
- Redistributions of source code must retain the above copyright | |
notice, this list of conditions and the following disclaimer. | |
- Redistributions in binary form must reproduce the above copyright | |
notice, this list of conditions and the following disclaimer in the | |
documentation and/or other materials provided with the distribution. | |
- Neither the name of the University nor the names of its contributors | |
may be used to endorse or promote products derived from this software | |
without specific prior written permission. | |
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | |
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | |
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | |
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE | |
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR | |
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | |
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | |
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN | |
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) | |
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | |
POSSIBILITY OF SUCH DAMAGE. | |
NetBSD: setvbuf.c,v 1.17 2003/08/07 16:43:31 agc Exp | |
setvbuf.c 8.2 (Berkeley) 11/16/93 | |
**/ | |
#include <LibConfig.h> | |
#include <assert.h> | |
#include <errno.h> | |
#include <stdio.h> | |
#include <stdlib.h> | |
#include <wchar.h> | |
#include "reentrant.h" | |
#include "local.h" | |
#include <MainData.h> | |
/* | |
* Set one of the three kinds of buffering, optionally including | |
* a buffer. | |
*/ | |
int | |
setvbuf(FILE *fp, char *buf, int mode, size_t size) | |
{ | |
int ret, flags; | |
size_t iosize; | |
int ttyflag; | |
_DIAGASSERT(fp != NULL); | |
/* buf may be NULL */ | |
if(fp == NULL) { | |
errno = EINVAL; | |
return (EOF); | |
} | |
/* | |
* Verify arguments. The `int' limit on `size' is due to this | |
* particular implementation. Note, buf and size are ignored | |
* when setting _IONBF. | |
*/ | |
if (mode != _IONBF) | |
if ((mode != _IOFBF && mode != _IOLBF) || (int)size < 0) | |
return (-1); | |
FLOCKFILE(fp); | |
/* | |
* Write current buffer, if any. Discard unread input (including | |
* ungetc data), cancel line buffering, and free old buffer if | |
* malloc()ed. We also clear any eof condition, as if this were | |
* a seek. | |
*/ | |
ret = 0; | |
(void)__sflush(fp); | |
if (HASUB(fp)) | |
FREEUB(fp); | |
WCIO_FREE(fp); | |
fp->_r = fp->_lbfsize = 0; | |
flags = fp->_flags; | |
if (flags & __SMBF) | |
free((void *)fp->_bf._base); | |
flags &= ~(__SLBF | __SNBF | __SMBF | __SOPT | __SNPT | __SEOF); | |
/* If setting unbuffered mode, skip all the hard work. */ | |
if (mode == _IONBF) | |
goto nbf; | |
/* | |
* Find optimal I/O size for seek optimization. This also returns | |
* a `tty flag' to suggest that we check isatty(fd), but we do not | |
* care since our caller told us how to buffer. | |
*/ | |
flags |= __swhatbuf(fp, &iosize, &ttyflag); | |
if (size == 0) { | |
buf = NULL; /* force local allocation */ | |
size = iosize; | |
} | |
/* Allocate buffer if needed. */ | |
if (buf == NULL) { | |
if ((buf = malloc(size)) == NULL) { | |
/* | |
* Unable to honor user's request. We will return | |
* failure, but try again with file system size. | |
*/ | |
ret = -1; | |
if (size != iosize) { | |
size = iosize; | |
buf = malloc(size); | |
} | |
} | |
if (buf == NULL) { | |
/* No luck; switch to unbuffered I/O. */ | |
nbf: | |
fp->_flags = (unsigned short)(flags | __SNBF); | |
fp->_w = 0; | |
fp->_bf._base = fp->_p = fp->_nbuf; | |
fp->_bf._size = 1; | |
FUNLOCKFILE(fp); | |
return (ret); | |
} | |
flags |= __SMBF; | |
} | |
/* | |
* Kill any seek optimization if the buffer is not the | |
* right size. | |
* | |
* SHOULD WE ALLOW MULTIPLES HERE (i.e., ok iff (size % iosize) == 0)? | |
*/ | |
if (size != iosize) | |
flags |= __SNPT; | |
/* | |
* Fix up the FILE fields, and set gMD->cleanup for output flush on | |
* exit (since we are buffered in some way). | |
*/ | |
if (mode == _IOLBF) | |
flags |= __SLBF; | |
fp->_flags = (unsigned short)flags; | |
fp->_bf._base = fp->_p = (unsigned char *)buf; | |
fp->_bf._size = (int)size; | |
/* fp->_lbfsize is still 0 */ | |
if (flags & __SWR) { | |
/* | |
* Begin or continue writing: see __swsetup(). Note | |
* that __SNBF is impossible (it was handled earlier). | |
*/ | |
if (flags & __SLBF) { | |
fp->_w = 0; | |
fp->_lbfsize = -fp->_bf._size; | |
} else | |
fp->_w = (int)size; | |
} else { | |
/* begin/continue reading, or stay in intermediate state */ | |
fp->_w = 0; | |
} | |
gMD->cleanup = _cleanup; | |
FUNLOCKFILE(fp); | |
return (ret); | |
} |