blob: 2c4b8f62f9445c1ff7995ae6ee7de0a143338ce2 [file] [log] [blame]
Jari Aalto31859422009-01-12 13:36:28 +00001/* zread - read data from file descriptor into buffer with retries */
2
Jari Aalto7117c2d2002-07-17 14:10:11 +00003/* Copyright (C) 1999-2002 Free Software Foundation, Inc.
Jari Aaltobb706242000-03-17 21:46:59 +00004
5 This file is part of GNU Bash, the Bourne Again SHell.
6
Jari Aalto31859422009-01-12 13:36:28 +00007 Bash is free software: you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation, either version 3 of the License, or
10 (at your option) any later version.
Jari Aaltobb706242000-03-17 21:46:59 +000011
Jari Aalto31859422009-01-12 13:36:28 +000012 Bash is distributed in the hope that it will be useful,
13 but 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 Bash. If not, see <http://www.gnu.org/licenses/>.
19*/
Jari Aaltobb706242000-03-17 21:46:59 +000020
21#include <config.h>
22
23#include <sys/types.h>
24
25#if defined (HAVE_UNISTD_H)
26# include <unistd.h>
27#endif
28
Chet Rameyac50fba2014-02-26 09:36:43 -050029#include <signal.h>
Jari Aaltobb706242000-03-17 21:46:59 +000030#include <errno.h>
31
32#if !defined (errno)
Ricardo Cerqueiraa02fbff2013-07-25 22:35:34 +010033#include <errno.h>
Jari Aaltobb706242000-03-17 21:46:59 +000034#endif
35
36#ifndef SEEK_CUR
37# define SEEK_CUR 1
38#endif
39
Chet Rameyac50fba2014-02-26 09:36:43 -050040extern void check_signals_and_traps (void);
41extern int signal_is_trapped (int);
42
Jari Aaltobb706242000-03-17 21:46:59 +000043/* Read LEN bytes from FD into BUF. Retry the read on EINTR. Any other
44 error causes the loop to break. */
Jari Aaltof73dda02001-11-13 17:56:06 +000045ssize_t
Jari Aaltobb706242000-03-17 21:46:59 +000046zread (fd, buf, len)
47 int fd;
48 char *buf;
49 size_t len;
50{
Jari Aaltof73dda02001-11-13 17:56:06 +000051 ssize_t r;
Jari Aaltobb706242000-03-17 21:46:59 +000052
Chet Rameyac50fba2014-02-26 09:36:43 -050053#if 0
54#if defined (HAVE_SIGINTERRUPT)
55 if (signal_is_trapped (SIGCHLD))
56 siginterrupt (SIGCHLD, 1);
57#endif
58#endif
59
Jari Aaltobb706242000-03-17 21:46:59 +000060 while ((r = read (fd, buf, len)) < 0 && errno == EINTR)
Chet Rameyac50fba2014-02-26 09:36:43 -050061 check_signals_and_traps (); /* XXX - should it be check_signals()? */
62
63#if 0
64#if defined (HAVE_SIGINTERRUPT)
65 siginterrupt (SIGCHLD, 0);
66#endif
67#endif
68
Jari Aaltobb706242000-03-17 21:46:59 +000069 return r;
70}
71
72/* Read LEN bytes from FD into BUF. Retry the read on EINTR, up to three
73 interrupts. Any other error causes the loop to break. */
74
75#ifdef NUM_INTR
76# undef NUM_INTR
77#endif
78#define NUM_INTR 3
79
Jari Aaltof73dda02001-11-13 17:56:06 +000080ssize_t
Jari Aalto17345e52009-02-19 22:21:29 +000081zreadretry (fd, buf, len)
Jari Aaltobb706242000-03-17 21:46:59 +000082 int fd;
83 char *buf;
84 size_t len;
85{
Jari Aaltof73dda02001-11-13 17:56:06 +000086 ssize_t r;
87 int nintr;
Jari Aaltobb706242000-03-17 21:46:59 +000088
89 for (nintr = 0; ; )
90 {
91 r = read (fd, buf, len);
92 if (r >= 0)
Jari Aalto28ef6c32001-04-06 19:14:31 +000093 return r;
Jari Aaltobb706242000-03-17 21:46:59 +000094 if (r == -1 && errno == EINTR)
95 {
Jari Aalto17345e52009-02-19 22:21:29 +000096 if (++nintr >= NUM_INTR)
Jari Aaltobb706242000-03-17 21:46:59 +000097 return -1;
98 continue;
99 }
100 return r;
101 }
102}
103
Jari Aalto17345e52009-02-19 22:21:29 +0000104/* Call read(2) and allow it to be interrupted. Just a stub for now. */
105ssize_t
106zreadintr (fd, buf, len)
107 int fd;
108 char *buf;
109 size_t len;
110{
111 return (read (fd, buf, len));
112}
113
Jari Aaltobb706242000-03-17 21:46:59 +0000114/* Read one character from FD and return it in CP. Return values are as
115 in read(2). This does some local buffering to avoid many one-character
116 calls to read(2), like those the `read' builtin performs. */
117
Jari Aaltof73dda02001-11-13 17:56:06 +0000118static char lbuf[128];
119static size_t lind, lused;
Jari Aaltobb706242000-03-17 21:46:59 +0000120
Jari Aaltof73dda02001-11-13 17:56:06 +0000121ssize_t
Jari Aaltobb706242000-03-17 21:46:59 +0000122zreadc (fd, cp)
123 int fd;
124 char *cp;
125{
Jari Aaltof73dda02001-11-13 17:56:06 +0000126 ssize_t nr;
Jari Aaltobb706242000-03-17 21:46:59 +0000127
128 if (lind == lused || lused == 0)
129 {
Jari Aaltof73dda02001-11-13 17:56:06 +0000130 nr = zread (fd, lbuf, sizeof (lbuf));
Jari Aaltobb706242000-03-17 21:46:59 +0000131 lind = 0;
Jari Aaltof73dda02001-11-13 17:56:06 +0000132 if (nr <= 0)
133 {
134 lused = 0;
135 return nr;
136 }
137 lused = nr;
Jari Aaltobb706242000-03-17 21:46:59 +0000138 }
139 if (cp)
Jari Aaltof73dda02001-11-13 17:56:06 +0000140 *cp = lbuf[lind++];
Jari Aaltobb706242000-03-17 21:46:59 +0000141 return 1;
142}
143
Jari Aalto17345e52009-02-19 22:21:29 +0000144/* Don't mix calls to zreadc and zreadcintr in the same function, since they
145 use the same local buffer. */
146ssize_t
147zreadcintr (fd, cp)
148 int fd;
149 char *cp;
150{
151 ssize_t nr;
152
153 if (lind == lused || lused == 0)
154 {
155 nr = zreadintr (fd, lbuf, sizeof (lbuf));
156 lind = 0;
157 if (nr <= 0)
158 {
159 lused = 0;
160 return nr;
161 }
162 lused = nr;
163 }
164 if (cp)
165 *cp = lbuf[lind++];
166 return 1;
167}
168
Chet Rameyac50fba2014-02-26 09:36:43 -0500169/* Like zreadc, but read a specified number of characters at a time. Used
170 for `read -N'. */
171ssize_t
172zreadn (fd, cp, len)
173 int fd;
174 char *cp;
175 size_t len;
176{
177 ssize_t nr;
178
179 if (lind == lused || lused == 0)
180 {
181 if (len > sizeof (lbuf))
182 len = sizeof (lbuf);
183 nr = zread (fd, lbuf, len);
184 lind = 0;
185 if (nr <= 0)
186 {
187 lused = 0;
188 return nr;
189 }
190 lused = nr;
191 }
192 if (cp)
193 *cp = lbuf[lind++];
194 return 1;
195}
196
Jari Aaltobb706242000-03-17 21:46:59 +0000197void
198zreset ()
199{
200 lind = lused = 0;
201}
202
203/* Sync the seek pointer for FD so that the kernel's idea of the last char
204 read is the last char returned by zreadc. */
205void
206zsyncfd (fd)
207 int fd;
208{
Chet Rameyea6616c2012-03-13 15:11:43 -0400209 off_t off, r;
Jari Aaltobb706242000-03-17 21:46:59 +0000210
211 off = lused - lind;
Jari Aalto95732b42005-12-07 14:08:12 +0000212 r = 0;
Jari Aaltobb706242000-03-17 21:46:59 +0000213 if (off > 0)
Jari Aalto95732b42005-12-07 14:08:12 +0000214 r = lseek (fd, -off, SEEK_CUR);
215
Chet Rameyea6616c2012-03-13 15:11:43 -0400216 if (r != -1)
Jari Aalto95732b42005-12-07 14:08:12 +0000217 lused = lind = 0;
Jari Aaltobb706242000-03-17 21:46:59 +0000218}