source: trunk/libs/newlib/src/newlib/libc/stdio/makebuf.c @ 577

Last change on this file since 577 was 444, checked in by satin@…, 6 years ago

add newlib,libalmos-mkh, restructure shared_syscalls.h and mini-libc

File size: 3.3 KB
Line 
1/*
2 * Copyright (c) 1990, 2007 The Regents of the University of California.
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms are permitted
6 * provided that the above copyright notice and this paragraph are
7 * duplicated in all such forms and that any documentation,
8 * advertising materials, and other materials related to such
9 * distribution and use acknowledge that the software was developed
10 * by the University of California, Berkeley.  The name of the
11 * University may not be used to endorse or promote products derived
12 * from this software without specific prior written permission.
13 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
14 * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
15 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
16 */
17/* No user fns here.  Pesch 15apr92. */
18
19#include <_ansi.h>
20#include <stdio.h>
21#include <stdlib.h>
22#include <sys/stat.h>
23#include <sys/types.h>
24#include <sys/unistd.h>
25#include "local.h"
26
27#define _DEFAULT_ASPRINTF_BUFSIZE 64
28
29/*
30 * Allocate a file buffer, or switch to unbuffered I/O.
31 * Per the ANSI C standard, ALL tty devices default to line buffered.
32 *
33 * As a side effect, we set __SOPT or __SNPT (en/dis-able fseek
34 * optimization) right after the _fstat() that finds the buffer size.
35 */
36
37void
38__smakebuf_r (struct _reent *ptr,
39       register FILE *fp)
40{
41  register void *p;
42  int flags;
43  size_t size;
44  int couldbetty;
45
46  if (fp->_flags & __SNBF)
47    {
48      fp->_bf._base = fp->_p = fp->_nbuf;
49      fp->_bf._size = 1;
50      return;
51    }
52  flags = __swhatbuf_r (ptr, fp, &size, &couldbetty);
53  if ((p = _malloc_r (ptr, size)) == NULL)
54    {
55      if (!(fp->_flags & __SSTR))
56        {
57          fp->_flags = (fp->_flags & ~__SLBF) | __SNBF;
58          fp->_bf._base = fp->_p = fp->_nbuf;
59          fp->_bf._size = 1;
60        }
61    }
62  else
63    {
64      ptr->__cleanup = _cleanup_r;
65      fp->_flags |= __SMBF;
66      fp->_bf._base = fp->_p = (unsigned char *) p;
67      fp->_bf._size = size;
68      if (couldbetty && _isatty_r (ptr, fp->_file))
69        fp->_flags = (fp->_flags & ~__SNBF) | __SLBF;
70      fp->_flags |= flags;
71    }
72}
73
74/*
75 * Internal routine to determine `proper' buffering for a file.
76 */
77int
78__swhatbuf_r (struct _reent *ptr,
79        FILE *fp,
80        size_t *bufsize,
81        int *couldbetty)
82{
83#ifdef _FSEEK_OPTIMIZATION
84  const int snpt = __SNPT;
85#else
86  const int snpt = 0;
87#endif
88
89#ifdef __USE_INTERNAL_STAT64
90  struct stat64 st;
91
92  if (fp->_file < 0 || _fstat64_r (ptr, fp->_file, &st) < 0)
93#else
94  struct stat st;
95
96  if (fp->_file < 0 || _fstat_r (ptr, fp->_file, &st) < 0)
97#endif
98    {
99      *couldbetty = 0;
100      /* Check if we are be called by asprintf family for initial buffer.  */
101      if (fp->_flags & __SMBF)
102        *bufsize = _DEFAULT_ASPRINTF_BUFSIZE;
103      else
104        *bufsize = BUFSIZ;
105      return (0);
106    }
107
108  /* could be a tty iff it is a character device */
109  *couldbetty = S_ISCHR(st.st_mode);
110#ifdef HAVE_BLKSIZE
111  if (st.st_blksize > 0)
112    {
113      /*
114       * Optimise fseek() only if it is a regular file.  (The test for
115       * __sseek is mainly paranoia.)  It is safe to set _blksize
116       * unconditionally; it will only be used if __SOPT is also set.
117       */
118      *bufsize = st.st_blksize;
119      fp->_blksize = st.st_blksize;
120      return ((st.st_mode & S_IFMT) == S_IFREG ?  __SOPT : snpt);
121    }
122#endif
123  *bufsize = BUFSIZ;
124  return (snpt);
125}
Note: See TracBrowser for help on using the repository browser.