mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-01-15 19:09:58 +08:00
05b315770a
* libc/stdio/asprintf.c libc/stdio/clearerr.c, libc/stdio/fclose.c libc/stdio/fcloseall.c libc/stdio/fdopen.c, libc/stdio/feof.c libc/stdio/ferror.c libc/stdio/fflush.c, libc/stdio/fgetc.c libc/stdio/fgetpos.c libc/stdio/fgets.c, libc/stdio/fileno.c libc/stdio/findfp.c libc/stdio/fiprintf.c, libc/stdio/flags.c libc/stdio/fopen.c libc/stdio/fprintf.c, libc/stdio/fputc.c libc/stdio/fputs.c libc/stdio/fread.c, libc/stdio/freopen.c libc/stdio/fscanf.c libc/stdio/fseek.c, libc/stdio/fseeko.c libc/stdio/fsetpos.c libc/stdio/ftell.c, libc/stdio/ftello.c libc/stdio/fvwrite.c libc/stdio/fwalk.c, libc/stdio/fwrite.c libc/stdio/getc.c libc/stdio/getc_u.c, libc/stdio/getchar.c libc/stdio/getchar_u.c, libc/stdio/getdelim.c libc/stdio/getline.c libc/stdio/gets.c, libc/stdio/getw.c libc/stdio/iprintf.c libc/stdio/local.h, libc/stdio/makebuf.c libc/stdio/mktemp.c libc/stdio/perror.c, libc/stdio/printf.c libc/stdio/putc.c libc/stdio/putc_u.c, libc/stdio/putchar.c libc/stdio/putchar_u.c libc/stdio/puts.c, libc/stdio/putw.c libc/stdio/refill.c libc/stdio/remove.c, libc/stdio/rename.c libc/stdio/rewind.c libc/stdio/rget.c, libc/stdio/scanf.c libc/stdio/setbuf.c libc/stdio/setbuffer.c, libc/stdio/setlinebuf.c libc/stdio/setvbuf.c, libc/stdio/siprintf.c libc/stdio/snprintf.c, libc/stdio/sprintf.c libc/stdio/sscanf.c libc/stdio/stdio.c, libc/stdio/tmpfile.c libc/stdio/tmpnam.c libc/stdio/ungetc.c, libc/stdio/vasprintf.c libc/stdio/vfieeefp.h, libc/stdio/vfprintf.c libc/stdio/vfscanf.c, libc/stdio/vprintf.c libc/stdio/vscanf.c, libc/stdio/vsnprintf.c libc/stdio/vsprintf.c, libc/stdio/vsscanf.c libc/stdio/wbuf.c, libc/stdio/wsetup.c: Perform minor formatting changes. Move copyright notices to top of file, ensure that <_ansi.h> is included, be consistent with open parentheses, use _DEFUN macro, include "local.h" where needed, and remove various compiler warnings.
249 lines
5.3 KiB
C
249 lines
5.3 KiB
C
/*
|
|
* Copyright (c) 1990 The Regents of the University of California.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms are permitted
|
|
* provided that the above copyright notice and this paragraph are
|
|
* duplicated in all such forms and that any documentation,
|
|
* advertising materials, and other materials related to such
|
|
* distribution and use acknowledge that the software was developed
|
|
* by the University of California, Berkeley. The name of the
|
|
* University may not be used to endorse or promote products derived
|
|
* from this software without specific prior written permission.
|
|
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
|
|
*/
|
|
/* No user fns here. Pesch 15apr92. */
|
|
|
|
#include <_ansi.h>
|
|
#include <reent.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#include <fcntl.h>
|
|
#include <sys/lock.h>
|
|
#include "local.h"
|
|
|
|
static _VOID
|
|
_DEFUN(std, (ptr, flags, file, data),
|
|
FILE *ptr _AND
|
|
int flags _AND
|
|
int file _AND
|
|
struct _reent *data)
|
|
{
|
|
ptr->_p = 0;
|
|
ptr->_r = 0;
|
|
ptr->_w = 0;
|
|
ptr->_flags = flags;
|
|
ptr->_file = file;
|
|
ptr->_bf._base = 0;
|
|
ptr->_bf._size = 0;
|
|
ptr->_lbfsize = 0;
|
|
ptr->_cookie = ptr;
|
|
ptr->_read = __sread;
|
|
ptr->_write = __swrite;
|
|
ptr->_seek = __sseek;
|
|
ptr->_close = __sclose;
|
|
#if !defined(__SINGLE_THREAD__) && !defined(_REENT_SMALL)
|
|
__lock_init_recursive (*(_LOCK_RECURSIVE_T *)&ptr->_lock);
|
|
/*
|
|
* #else
|
|
* lock is already initialized in __sfp
|
|
*/
|
|
#endif
|
|
|
|
#ifdef __SCLE
|
|
if (__stextmode (ptr->_file))
|
|
ptr->_flags |= __SCLE;
|
|
#endif
|
|
}
|
|
|
|
struct _glue *
|
|
_DEFUN(__sfmoreglue, (d, n),
|
|
struct _reent *d _AND
|
|
register int n)
|
|
{
|
|
struct _glue *g;
|
|
FILE *p;
|
|
|
|
g = (struct _glue *) _malloc_r (d, sizeof (*g) + n * sizeof (FILE));
|
|
if (g == NULL)
|
|
return NULL;
|
|
p = (FILE *) (g + 1);
|
|
g->_next = NULL;
|
|
g->_niobs = n;
|
|
g->_iobs = p;
|
|
memset (p, 0, n * sizeof (FILE));
|
|
return g;
|
|
}
|
|
|
|
/*
|
|
* Find a free FILE for fopen et al.
|
|
*/
|
|
|
|
FILE *
|
|
_DEFUN(__sfp, (d),
|
|
struct _reent *d)
|
|
{
|
|
FILE *fp;
|
|
int n;
|
|
struct _glue *g;
|
|
|
|
__sfp_lock_acquire ();
|
|
|
|
if (!_GLOBAL_REENT->__sdidinit)
|
|
__sinit (_GLOBAL_REENT);
|
|
for (g = &_GLOBAL_REENT->__sglue;; g = g->_next)
|
|
{
|
|
for (fp = g->_iobs, n = g->_niobs; --n >= 0; fp++)
|
|
if (fp->_flags == 0)
|
|
goto found;
|
|
if (g->_next == NULL &&
|
|
(g->_next = __sfmoreglue (d, NDYNAMIC)) == NULL)
|
|
break;
|
|
}
|
|
__sfp_lock_release ();
|
|
d->_errno = ENOMEM;
|
|
return NULL;
|
|
|
|
found:
|
|
fp->_file = -1; /* no file */
|
|
fp->_flags = 1; /* reserve this slot; caller sets real flags */
|
|
#ifndef __SINGLE_THREAD__
|
|
__lock_init_recursive (*(_LOCK_RECURSIVE_T *)&fp->_lock);
|
|
#endif
|
|
__sfp_lock_release ();
|
|
|
|
fp->_p = NULL; /* no current pointer */
|
|
fp->_w = 0; /* nothing to read or write */
|
|
fp->_r = 0;
|
|
fp->_bf._base = NULL; /* no buffer */
|
|
fp->_bf._size = 0;
|
|
fp->_lbfsize = 0; /* not line buffered */
|
|
/* fp->_cookie = <any>; */ /* caller sets cookie, _read/_write etc */
|
|
fp->_ub._base = NULL; /* no ungetc buffer */
|
|
fp->_ub._size = 0;
|
|
fp->_lb._base = NULL; /* no line buffer */
|
|
fp->_lb._size = 0;
|
|
|
|
return fp;
|
|
}
|
|
|
|
/*
|
|
* exit() calls _cleanup() through *__cleanup, set whenever we
|
|
* open or buffer a file. This chicanery is done so that programs
|
|
* that do not use stdio need not link it all in.
|
|
*
|
|
* The name `_cleanup' is, alas, fairly well known outside stdio.
|
|
*/
|
|
|
|
_VOID
|
|
_DEFUN(_cleanup_r, (ptr),
|
|
struct _reent *ptr)
|
|
{
|
|
/* _CAST_VOID _fwalk(fclose); */
|
|
_CAST_VOID _fwalk (ptr, fflush); /* `cheating' */
|
|
}
|
|
|
|
#ifndef _REENT_ONLY
|
|
_VOID
|
|
_DEFUN_VOID(_cleanup)
|
|
{
|
|
_cleanup_r (_GLOBAL_REENT);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* __sinit() is called whenever stdio's internal variables must be set up.
|
|
*/
|
|
|
|
_VOID
|
|
_DEFUN(__sinit, (s),
|
|
struct _reent *s)
|
|
{
|
|
/* make sure we clean up on exit */
|
|
s->__cleanup = _cleanup_r; /* conservative */
|
|
s->__sdidinit = 1;
|
|
|
|
s->__sglue._next = NULL;
|
|
#ifndef _REENT_SMALL
|
|
s->__sglue._niobs = 3;
|
|
s->__sglue._iobs = &s->__sf[0];
|
|
#else
|
|
s->__sglue._niobs = 0;
|
|
s->__sglue._iobs = NULL;
|
|
s->_stdin = __sfp(s);
|
|
s->_stdout = __sfp(s);
|
|
s->_stderr = __sfp(s);
|
|
#endif
|
|
|
|
std (s->_stdin, __SRD, 0, s);
|
|
|
|
/* on platforms that have true file system I/O, we can verify whether stdout
|
|
is an interactive terminal or not. For all other platforms, we will
|
|
default to line buffered mode here. */
|
|
#ifdef HAVE_FCNTL
|
|
std (s->_stdout, __SWR, 1, s);
|
|
#else
|
|
std (s->_stdout, __SWR | __SLBF, 1, s);
|
|
#endif
|
|
|
|
std (s->_stderr, __SWR | __SNBF, 2, s);
|
|
|
|
}
|
|
|
|
#ifndef __SINGLE_THREAD__
|
|
|
|
__LOCK_INIT_RECURSIVE(static, __sfp_lock);
|
|
|
|
_VOID
|
|
_DEFUN_VOID(__sfp_lock_acquire)
|
|
{
|
|
__lock_acquire_recursive (__sfp_lock);
|
|
}
|
|
|
|
_VOID
|
|
_DEFUN_VOID(__sfp_lock_release)
|
|
{
|
|
__lock_release_recursive (__sfp_lock);
|
|
}
|
|
|
|
/* Walkable file locking routine. */
|
|
static int
|
|
_DEFUN(__fp_lock, (ptr),
|
|
FILE * ptr)
|
|
{
|
|
_flockfile (ptr);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Walkable file unlocking routine. */
|
|
static int
|
|
_DEFUN(__fp_unlock, (ptr),
|
|
FILE * ptr)
|
|
{
|
|
_funlockfile (ptr);
|
|
|
|
return 0;
|
|
}
|
|
|
|
_VOID
|
|
_DEFUN_VOID(__fp_lock_all)
|
|
{
|
|
__sfp_lock_acquire ();
|
|
|
|
_CAST_VOID _fwalk (_REENT, __fp_lock);
|
|
}
|
|
|
|
_VOID
|
|
_DEFUN_VOID(__fp_unlock_all)
|
|
{
|
|
_CAST_VOID _fwalk (_REENT, __fp_unlock);
|
|
|
|
__sfp_lock_release ();
|
|
}
|
|
#endif
|