kernel - Video - Add suppor for Intel IGD chipsets (netbook / N450 etc)
[dragonfly.git] / lib / libc / stdio / refill.c
blobda51bf5e2e7b24fc2326d0547cb1b7404ffa21b8
1 /*-
2 * Copyright (c) 1990, 1993
3 * The Regents of the University of California. All rights reserved.
5 * This code is derived from software contributed to Berkeley by
6 * Chris Torek.
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
10 * are met:
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
16 * 4. Neither the name of the University nor the names of its contributors
17 * may be used to endorse or promote products derived from this software
18 * without specific prior written permission.
20 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
21 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
24 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30 * SUCH DAMAGE.
32 * @(#)refill.c 8.1 (Berkeley) 6/4/93
33 * $FreeBSD: src/lib/libc/stdio/refill.c,v 1.20 2008/04/17 22:17:54 jhb Exp $
34 * $DragonFly: src/lib/libc/stdio/refill.c,v 1.10 2005/11/20 11:07:30 swildner Exp $
37 #include "namespace.h"
38 #include <errno.h>
39 #include <stdio.h>
40 #include <stdlib.h>
41 #include "un-namespace.h"
43 #include "libc_private.h"
44 #include "local.h"
45 #include "priv_stdio.h"
47 static int lflush(FILE *);
49 static int
50 lflush(FILE *fp)
52 int ret = 0;
54 if ((fp->pub._flags & (__SLBF|__SWR)) == (__SLBF|__SWR)) {
55 FLOCKFILE(fp);
56 ret = __sflush(fp);
57 FUNLOCKFILE(fp);
59 return (ret);
63 * Refill a stdio buffer.
64 * Return EOF on eof or error, 0 otherwise.
66 int
67 __srefill(FILE *fp)
70 /* make sure stdio is set up */
71 if (!__sdidinit)
72 __sinit();
74 ORIENT(fp, -1);
76 fp->pub._r = 0; /* largely a convenience for callers */
78 /* SysV does not make this test; take it out for compatibility */
79 if (fp->pub._flags & __SEOF)
80 return (EOF);
82 /* if not already reading, have to be reading and writing */
83 if ((fp->pub._flags & __SRD) == 0) {
84 if ((fp->pub._flags & __SRW) == 0) {
85 errno = EBADF;
86 fp->pub._flags |= __SERR;
87 return (EOF);
89 /* switch to reading */
90 if (fp->pub._flags & __SWR) {
91 if (__sflush(fp))
92 return (EOF);
93 fp->pub._flags &= ~__SWR;
94 fp->pub._w = 0;
95 fp->pub._lbfsize = 0;
97 fp->pub._flags |= __SRD;
98 } else {
100 * We were reading. If there is an ungetc buffer,
101 * we must have been reading from that. Drop it,
102 * restoring the previous buffer (if any). If there
103 * is anything in that buffer, return.
105 if (HASUB(fp)) {
106 FREEUB(fp);
107 if ((fp->pub._r = fp->_ur) != 0) {
108 fp->pub._p = fp->_up;
109 return (0);
114 if (fp->_bf._base == NULL)
115 __smakebuf(fp);
118 * Before reading from a line buffered or unbuffered file,
119 * flush all line buffered output files, per the ANSI C
120 * standard.
122 if (fp->pub._flags & (__SLBF|__SNBF)) {
123 /* Ignore this file in _fwalk to avoid potential deadlock. */
124 fp->pub._flags |= __SIGN;
125 _fwalk(lflush);
126 fp->pub._flags &= ~__SIGN;
128 /* Now flush this file without locking it. */
129 if ((fp->pub._flags & (__SLBF|__SWR)) == (__SLBF|__SWR))
130 __sflush(fp);
132 fp->pub._p = fp->_bf._base;
133 fp->pub._r = _sread(fp, (char *)fp->pub._p, fp->_bf._size);
134 fp->pub._flags &= ~__SMOD; /* buffer contents are again pristine */
135 if (fp->pub._r <= 0) {
136 if (fp->pub._r == 0)
137 fp->pub._flags |= __SEOF;
138 else {
139 fp->pub._r = 0;
140 fp->pub._flags |= __SERR;
142 return (EOF);
144 return (0);