blob: 3bf0c2070eea684b5e8855f2a3d1a2f2050a6a4a [file] [log] [blame]
/* Output stream referring to a file descriptor.
Copyright (C) 2006-2007, 2019-2020 Free Software Foundation, Inc.
Written by Bruno Haible <bruno@clisp.org>, 2006.
This program 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.
This program 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 this program. If not, see <https://www.gnu.org/licenses/>. */
#include <config.h>
/* Specification. */
#include "fd-ostream.h"
#include <assert.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#if HAVE_TCDRAIN
# include <termios.h>
#endif
#include "error.h"
#include "full-write.h"
#include "xalloc.h"
#include "gettext.h"
#define _(str) gettext (str)
struct fd_ostream : struct ostream
{
fields:
int fd;
char *filename;
char *buffer; /* A buffer, or NULL. */
size_t avail; /* Number of bytes available in the buffer. */
};
#define BUFSIZE 4096
#if HAVE_TCDRAIN
/* EINTR handling for tcdrain().
This function can return -1/EINTR even though we don't have any
signal handlers set up, namely when we get interrupted via SIGSTOP. */
static inline int
nonintr_tcdrain (int fd)
{
int retval;
do
retval = tcdrain (fd);
while (retval < 0 && errno == EINTR);
return retval;
}
#endif
/* Implementation of ostream_t methods. */
static void
fd_ostream::write_mem (fd_ostream_t stream, const void *data, size_t len)
{
if (len > 0)
{
if (stream->buffer != NULL)
{
/* Buffered. */
assert (stream->avail > 0);
#if 0 /* unoptimized */
do
{
size_t n = (len <= stream->avail ? len : stream->avail);
if (n > 0)
{
memcpy (stream->buffer + BUFSIZE - stream->avail, data, n);
data = (const char *) data + n;
stream->avail -= n;
len -= n;
}
if (stream->avail == 0)
{
if (full_write (stream->fd, stream->buffer, BUFSIZE) < BUFSIZE)
error (EXIT_FAILURE, errno, _("error writing to %s"),
stream->filename);
stream->avail = BUFSIZE;
}
}
while (len > 0);
#else /* optimized */
if (len < stream->avail)
{
/* Move the data into the buffer. */
memcpy (stream->buffer + BUFSIZE - stream->avail, data, len);
stream->avail -= len;
}
else
{
/* Split the data into:
- a first chunk, which is added to the buffer and output,
- a series of chunks of size BUFSIZE, which can be output
directly, without going through the buffer, and
- a last chunk, which is copied to the buffer. */
size_t n = stream->avail;
memcpy (stream->buffer + BUFSIZE - stream->avail, data, n);
data = (const char *) data + n;
len -= n;
if (full_write (stream->fd, stream->buffer, BUFSIZE) < BUFSIZE)
error (EXIT_FAILURE, errno, _("error writing to %s"),
stream->filename);
while (len >= BUFSIZE)
{
if (full_write (stream->fd, data, BUFSIZE) < BUFSIZE)
error (EXIT_FAILURE, errno, _("error writing to %s"),
stream->filename);
data = (const char *) data + BUFSIZE;
len -= BUFSIZE;
}
if (len > 0)
memcpy (stream->buffer, data, len);
stream->avail = BUFSIZE - len;
}
#endif
assert (stream->avail > 0);
}
else
{
/* Unbuffered. */
if (full_write (stream->fd, data, len) < len)
error (EXIT_FAILURE, errno, _("error writing to %s"),
stream->filename);
}
}
}
static void
fd_ostream::flush (fd_ostream_t stream, ostream_flush_scope_t scope)
{
if (stream->buffer != NULL && stream->avail < BUFSIZE)
{
size_t filled = BUFSIZE - stream->avail;
if (full_write (stream->fd, stream->buffer, filled) < filled)
error (EXIT_FAILURE, errno, _("error writing to %s"), stream->filename);
stream->avail = BUFSIZE;
}
if (scope == FLUSH_ALL)
{
/* For streams connected to a disk file: */
fsync (stream->fd);
#if HAVE_TCDRAIN
/* For streams connected to a terminal: */
nonintr_tcdrain (stream->fd);
#endif
}
}
static void
fd_ostream::free (fd_ostream_t stream)
{
fd_ostream_flush (stream, FLUSH_THIS_STREAM);
free (stream->filename);
free (stream);
}
/* Constructor. */
fd_ostream_t
fd_ostream_create (int fd, const char *filename, bool buffered)
{
fd_ostream_t stream =
(struct fd_ostream_representation *)
xmalloc (sizeof (struct fd_ostream_representation)
+ (buffered ? BUFSIZE : 0));
stream->base.vtable = &fd_ostream_vtable;
stream->fd = fd;
stream->filename = xstrdup (filename);
if (buffered)
{
stream->buffer =
(char *) (void *) stream + sizeof (struct fd_ostream_representation);
stream->avail = BUFSIZE;
}
else
stream->buffer = NULL;
return stream;
}
/* Accessors. */
static int
fd_ostream::get_descriptor (fd_ostream_t stream)
{
return stream->fd;
}
static const char *
fd_ostream::get_filename (fd_ostream_t stream)
{
return stream->filename;
}
static bool
fd_ostream::is_buffered (fd_ostream_t stream)
{
return stream->buffer != NULL;
}
/* Instanceof test. */
bool
is_instance_of_fd_ostream (ostream_t stream)
{
return IS_INSTANCE (stream, ostream, fd_ostream);
}