8dfa256c07f633dd1e4371c8d0d0333f10a4ff71
[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 #define __UNGETC_MAX 1024
28
29 int __ungetc_buf[__UNGETC_MAX + 1] = { 0 };
30
31 int
32 __ungetc_p (int filedes)
33 {
34   return __ungetc_buf[filedes] >= 0;
35 }
36
37 void
38 __ungetc_init ()
39 {
40   if (__ungetc_buf[__UNGETC_MAX] == 0)
41     memset (__ungetc_buf, -1, (__UNGETC_MAX + 1) * sizeof (int));
42 }
43
44 void
45 __ungetc_clear (int filedes)
46 {
47   __ungetc_buf[filedes] = -1;
48 }
49
50 void
51 __ungetc_set (int filedes, int c)
52 {
53   __ungetc_buf[filedes] = c;
54 }
55
56 int
57 fdgetc (int fd)
58 {
59   __ungetc_init ();
60
61   char c;
62   int i = __ungetc_buf[fd];
63   if (i >= 0)
64     __ungetc_buf[fd] = -1;
65   else
66     {
67       int r = read (fd, &c, 1);
68       if (r < 1)
69         return -1;
70       i = (int) c;
71     }
72   if (i < 0)
73     i += 256;
74
75   return i;
76 }