mescc: Mes C Library: Add buffered read.
[mes.git] / lib / mes / fdgetc.c
1 /* -*-comment-start: "//";comment-end:""-*-
2  * GNU Mes --- Maxwell Equations of Software
3  * Copyright © 2016,2017,2018,2019 Jan (janneke) Nieuwenhuizen <janneke@gnu.org>
4  *
5  * This file is part of GNU Mes.
6  *
7  * GNU Mes is free software; you can redistribute it and/or modify it
8  * under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 3 of the License, or (at
10  * your option) any later version.
11  *
12  * GNU Mes is distributed in the hope that it will be useful, but
13  * WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with GNU Mes.  If not, see <http://www.gnu.org/licenses/>.
19  */
20
21 #include <mes/lib.h>
22 #include <limits.h>
23 #include <string.h>
24 #include <sys/resource.h>
25 #include <unistd.h>
26
27 int __ungetc_buf[__FILEDES_MAX + 1] = { 0 };
28
29 int
30 __ungetc_p (int filedes)
31 {
32   return __ungetc_buf[filedes] >= 0;
33 }
34
35 void
36 __ungetc_init ()
37 {
38   if (__ungetc_buf[__FILEDES_MAX] == 0)
39     memset (__ungetc_buf, -1, (__FILEDES_MAX + 1) * sizeof (int));
40 }
41
42 void
43 __ungetc_clear (int filedes)
44 {
45   __ungetc_buf[filedes] = -1;
46 }
47
48 void
49 __ungetc_set (int filedes, int c)
50 {
51   __ungetc_buf[filedes] = c;
52 }
53
54 int
55 fdgetc (int fd)
56 {
57   __ungetc_init ();
58
59   char c;
60   int i = __ungetc_buf[fd];
61   if (i >= 0)
62     __ungetc_buf[fd] = -1;
63   else
64     {
65       int r = read (fd, &c, 1);
66       if (r < 1)
67         return -1;
68       i = (int) c;
69     }
70   if (i < 0)
71     i += 256;
72
73   return i;
74 }