Files
2025-07-03 16:15:36 -04:00

309 lines
6.3 KiB
C

/* zread - read data from file descriptor into buffer with retries */
/* Copyright (C) 1999-2024 Free Software Foundation, Inc.
This file is part of GNU Bash, the Bourne Again SHell.
Bash is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
Bash is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with Bash. If not, see <http://www.gnu.org/licenses/>.
*/
#include <config.h>
#include <sys/types.h>
#if defined (HAVE_UNISTD_H)
# include <unistd.h>
#endif
#include <signal.h>
#include <errno.h>
#if !defined (errno)
extern int errno;
#endif
#ifndef SEEK_CUR
# define SEEK_CUR 1
#endif
#ifndef ZBUFSIZ
# define ZBUFSIZ 4096
#endif
#ifndef EOF
# define EOF -1
#endif
extern int executing_builtin, interrupt_state;
extern void check_signals_and_traps (void);
extern void check_signals (void);
extern int signal_is_trapped (int);
extern int read_builtin_timeout (int);
/* Forward declarations */
void zreset (void);
int zungetc (int);
/* Provide 16 bytes of pushback whether we are using read or zread. Only used
by the read builtin when reading invalid multibyte characters. */
#define ZPUSHSIZE 16
static size_t zpushind, zpopind;
static unsigned char zpushbuf[ZPUSHSIZE];
static unsigned char zbufchar;
static inline int
zbufpop(unsigned char *cp)
{
if (zpushind == zpopind)
return (0);
*cp = zpushbuf[zpopind++];
if (zpopind == zpushind)
zpopind = zpushind = 0; /* reset, buffer empty */
return 1;
}
static inline int
zbufpush(int c)
{
if (zpushind == ZPUSHSIZE - 1)
return 0;
zpushbuf[zpushind++] = c;
return 1;
}
/* Add C to the pushback buffer. Can't push back EOF */
int
zungetc (int c)
{
zbufpush (c);
return c;
}
/* Read LEN bytes from FD into BUF. Retry the read on EINTR. Any other
error causes the loop to break. */
ssize_t
zread (int fd, char *buf, size_t len)
{
ssize_t r;
check_signals (); /* check for signals before a blocking read */
/* If we pushed chars back, return the oldest one immediately */
if (zbufpop (&zbufchar))
{
*buf = zbufchar;
return 1;
}
/* should generalize into a mechanism where different parts of the shell can
`register' timeouts and have them checked here. */
while (((r = read_builtin_timeout (fd)) < 0 || (r = read (fd, buf, len)) < 0) &&
errno == EINTR)
{
int t;
t = errno;
/* XXX - bash-5.0 */
/* We check executing_builtin and run traps here for backwards compatibility */
if (executing_builtin)
{
if (interrupt_state)
zreset ();
check_signals_and_traps (); /* XXX - should it be check_signals()? */
}
else
check_signals ();
errno = t;
}
return r;
}
/* Read LEN bytes from FD into BUF. Retry the read on EINTR, up to three
interrupts. Any other error causes the loop to break. */
#ifdef NUM_INTR
# undef NUM_INTR
#endif
#define NUM_INTR 3
ssize_t
zreadretry (int fd, char *buf, size_t len)
{
ssize_t r;
int nintr;
/* If we pushed chars back, return the oldest one immediately */
if (zbufpop (&zbufchar))
{
*buf = zbufchar;
return 1;
}
for (nintr = 0; ; )
{
r = read (fd, buf, len);
if (r >= 0)
return r;
if (r == -1 && errno == EINTR)
{
if (++nintr >= NUM_INTR)
return -1;
continue;
}
return r;
}
}
/* Call read(2) and allow it to be interrupted. Just a stub for now. */
ssize_t
zreadintr (int fd, char *buf, size_t len)
{
check_signals ();
/* If we pushed chars back, return the oldest one immediately */
if (zbufpop (&zbufchar))
{
*buf = zbufchar;
return 1;
}
return (read (fd, buf, len));
}
/* Read one character from FD and return it in CP. Return values are as
in read(2). This does some local buffering to avoid many one-character
calls to read(2), like those the `read' builtin performs. */
static char lbuf[ZBUFSIZ];
static size_t lind, lused;
ssize_t
zreadc (int fd, char *cp)
{
ssize_t nr;
/* If we pushed chars back, return the oldest one immediately */
if (cp && zbufpop (&zbufchar))
{
*cp = zbufchar;
return 1;
}
if (lind == lused || lused == 0)
{
nr = zread (fd, lbuf, sizeof (lbuf));
lind = 0;
if (nr <= 0)
{
lused = 0;
return nr;
}
lused = nr;
}
if (cp)
*cp = lbuf[lind++];
return 1;
}
/* Don't mix calls to zreadc and zreadcintr in the same function, since they
use the same local buffer. */
ssize_t
zreadcintr (int fd, char *cp)
{
ssize_t nr;
/* If we pushed chars back, return the oldest one immediately */
if (cp && zbufpop (&zbufchar))
{
*cp = zbufchar;
return 1;
}
if (lind == lused || lused == 0)
{
nr = zreadintr (fd, lbuf, sizeof (lbuf));
lind = 0;
if (nr <= 0)
{
lused = 0;
return nr;
}
lused = nr;
}
if (cp)
*cp = lbuf[lind++];
return 1;
}
/* Like zreadc, but read a specified number of characters at a time. Used
for `read -N'. */
ssize_t
zreadn (int fd, char *cp, size_t len)
{
ssize_t nr;
/* If we pushed chars back, return the oldest one immediately */
if (cp && zbufpop (&zbufchar))
{
*cp = zbufchar;
return 1;
}
if (lind == lused || lused == 0)
{
if (len > sizeof (lbuf))
len = sizeof (lbuf);
nr = zread (fd, lbuf, len);
lind = 0;
if (nr <= 0)
{
lused = 0;
return nr;
}
lused = nr;
}
if (cp)
*cp = lbuf[lind++];
return 1;
}
void
zreset (void)
{
lind = lused = 0;
zpushind = zpopind = 0;
}
/* Sync the seek pointer for FD so that the kernel's idea of the last char
read is the last char returned by zreadc. */
void
zsyncfd (int fd)
{
off_t off, r;
off = lused - lind;
r = 0;
if (off > 0)
r = lseek (fd, -off, SEEK_CUR);
if (r != -1)
{
lused = lind = 0;
zpushind = zpopind = 0;
}
}