klayout/src/tl/tl/tlStream.cc

1601 lines
34 KiB
C++

/*
KLayout Layout Viewer
Copyright (C) 2006-2024 Matthias Koefferlein
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 2 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, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
#include <stddef.h>
#include <ctype.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <errno.h>
#include <zlib.h>
#ifdef _WIN32
# include <io.h>
#else
# include <unistd.h>
#endif
#include "tlStream.h"
#include "tlHttpStream.h"
#include "tlDeflate.h"
#include "tlAssert.h"
#include "tlFileUtils.h"
#include "tlLog.h"
#include "tlResources.h"
#include "tlBase64.h"
#include "tlException.h"
#include "tlString.h"
#include "tlUri.h"
#if defined(HAVE_QT)
# include <QByteArray>
# include <QResource>
#endif
namespace tl
{
// ---------------------------------------------------------------------------------
// Some exception classes
class FileWriteErrorException
: public tl::Exception
{
public:
FileWriteErrorException (const std::string &f, int en)
: tl::Exception (tl::to_string (tr ("Write error on file: %s (errno=%d)")), f, en)
{ }
};
class FileReadErrorException
: public tl::Exception
{
public:
FileReadErrorException (const std::string &f, int en)
: tl::Exception (tl::to_string (tr ("Read error on file: %s (errno=%d)")), f, en)
{ }
};
class ZLibWriteErrorException
: public tl::Exception
{
public:
ZLibWriteErrorException (const std::string &f, const char *em)
: tl::Exception (tl::to_string (tr ("Write error on file in decompression library: %s (message=%s)")), f, em)
{ }
};
class ZLibReadErrorException
: public tl::Exception
{
public:
ZLibReadErrorException (const std::string &f, const char *em)
: tl::Exception (tl::to_string (tr ("Read error on file in decompression library: %s (message=%s)")), f, em)
{ }
};
class FileOpenErrorException
: public tl::Exception
{
public:
FileOpenErrorException (const std::string &f, int en)
: tl::Exception (tl::to_string (tr ("Unable to open file: %s (errno=%d)")), f, en)
{ }
};
class FilePOpenErrorException
: public tl::Exception
{
public:
FilePOpenErrorException (const std::string &f, int en)
: tl::Exception (tl::to_string (tr ("Unable to get input from command through pipe: %s (errno=%d)")), f, en)
{ }
};
class FilePReadErrorException
: public tl::Exception
{
public:
FilePReadErrorException (const std::string &f, int en)
: tl::Exception (tl::to_string (tr ("Read error on pipe from command: %s (errno=%d)")), f, en)
{ }
};
class FilePWriteErrorException
: public tl::Exception
{
public:
FilePWriteErrorException (const std::string &f, int en)
: tl::Exception (tl::to_string (tr ("Write error on pipe from command: %s (errno=%d)")), f, en)
{ }
};
// ---------------------------------------------------------------
class ZLibFilePrivate
{
public:
ZLibFilePrivate () : zs (NULL) { }
gzFile zs;
};
// ---------------------------------------------------------------
// inflating_input_stream implementation
/**
* @brief A wrapper that adds generic .gz support
*/
template <class Base>
inflating_input_stream<Base>::inflating_input_stream (Base *delegate)
: m_inflating_stream (delegate), m_is_compressed (false), mp_delegate (delegate)
{
enter_inflate ();
}
template <class Base>
size_t
inflating_input_stream<Base>::read (char *b, size_t n)
{
// TODO: this is somewhat inefficient, but we only use it for pipe and HTTP streams
size_t i = 0;
while (i < n) {
if (m_is_compressed || m_inflating_stream.blen () == 0) {
const char *read = m_inflating_stream.get (1);
if (! read) {
break;
}
*b++ = *read;
i += 1;
} else {
size_t nn = std::min (n - i, m_inflating_stream.blen ());
const char *read = m_inflating_stream.get (nn);
tl_assert (read != 0);
memcpy (b, read, nn);
b += nn;
i += nn;
}
}
return i;
}
template <class Base>
void
inflating_input_stream<Base>::enter_inflate ()
{
// identify and skip header for .gz file
if (auto_detect_gz ()) {
m_is_compressed = true;
m_inflating_stream.inflate (true /* stop after inflated block */);
} else {
m_inflating_stream.unget (m_inflating_stream.pos ());
}
}
template <class Base>
bool
inflating_input_stream<Base>::auto_detect_gz ()
{
std::string header = m_inflating_stream.read_all (10);
if (header.size () < 10) {
return false;
}
const unsigned char *header_data = (const unsigned char *) header.c_str ();
unsigned char flags = header_data[3];
if (header_data[0] != 0x1f || header_data[1] != 0x8b || header_data[2] != 0x08 || (flags & 0xe0) != 0) {
return false;
}
// .gz signature found
bool has_fhcrc = (flags & 0x02) != 0;
bool has_extra = (flags & 0x04) != 0;
bool has_fname = (flags & 0x08) != 0;
bool has_comment = (flags & 0x10) != 0;
if (has_extra) {
const unsigned char *xlen = (const unsigned char *) m_inflating_stream.get (2);
if (! xlen) {
throw tl::Exception (tl::to_string (tr ("Corrupt .gz header - missing XLEN field")));
}
const char *xdata = m_inflating_stream.get (size_t (xlen[0]) + (size_t (xlen[1]) << 8));
if (! xdata) {
throw tl::Exception (tl::to_string (tr ("Corrupt .gz header - missing EXTRA data")));
}
}
if (has_fname) {
const char *c;
while ((c = m_inflating_stream.get (1)) != 0 && *c)
;
if (! c) {
throw tl::Exception (tl::to_string (tr ("Corrupt .gz header - missing FNAME data trailing zero byte")));
}
}
if (has_comment) {
const char *c;
while ((c = m_inflating_stream.get (1)) != 0 && *c)
;
if (! c) {
throw tl::Exception (tl::to_string (tr ("Corrupt .gz header - missing COMMENT data trailing zero byte")));
}
}
if (has_fhcrc) {
const char *crc16 = m_inflating_stream.get (2);
if (! crc16) {
throw tl::Exception (tl::to_string (tr ("Corrupt .gz header - missing CRC16 data")));
}
}
return true;
}
// ---------------------------------------------------------------
// InputStream implementation
namespace {
/**
* @brief A dummy delegate to provide for the case of raw data stashed inside the stream itself
*/
class RawDataDelegate
: public InputStreamBase
{
public:
RawDataDelegate (const std::string &source)
: m_source (source)
{ }
virtual size_t read (char *, size_t)
{
return 0;
}
virtual void reset () { }
virtual void close () { }
virtual std::string source () const { return m_source; }
virtual std::string absolute_path () const { return m_source; }
virtual std::string filename () const { return m_source; }
public:
std::string m_source;
};
}
InputStream::InputStream (InputStreamBase &delegate)
: m_pos (0), mp_bptr (0), mp_delegate (&delegate), m_owns_delegate (false), mp_inflate (0), m_inflate_always (false), m_stop_after_inflate (false)
{
m_bcap = 4096; // initial buffer capacity
m_blen = 0;
mp_buffer = new char [m_bcap];
}
InputStream::InputStream (InputStreamBase *delegate)
: m_pos (0), mp_bptr (0), mp_delegate (delegate), m_owns_delegate (true), mp_inflate (0), m_inflate_always (false), m_stop_after_inflate (false)
{
m_bcap = 4096; // initial buffer capacity
m_blen = 0;
mp_buffer = new char [m_bcap];
}
InputStream::InputStream (const std::string &abstract_path)
: m_pos (0), mp_bptr (0), mp_delegate (0), m_owns_delegate (false), mp_inflate (0), m_inflate_always (false), m_stop_after_inflate (false)
{
m_bcap = 4096; // initial buffer capacity
m_blen = 0;
mp_buffer = 0;
bool needs_inflate = false;
tl::Extractor ex (abstract_path.c_str ());
if (ex.test (":")) {
#if defined(HAVE_QT)
QResource res (tl::to_qstring (abstract_path));
if (res.size () == 0) {
throw tl::Exception (tl::to_string (tr ("Resource not found: ")) + abstract_path);
}
QByteArray data;
#if QT_VERSION >= 0x60000
if (res.compressionAlgorithm () == QResource::ZlibCompression) {
#else
if (res.isCompressed ()) {
#endif
data = qUncompress ((const unsigned char *)res.data (), (int)res.size ());
} else {
data = QByteArray ((const char *)res.data (), (int)res.size ());
}
mp_buffer = new char[data.size ()];
memcpy (mp_buffer, data.constData (), data.size ());
mp_bptr = mp_buffer;
m_bcap = data.size ();
m_blen = m_bcap;
mp_delegate = new RawDataDelegate (abstract_path);
#else
std::pair<tl::InputStreamBase *, bool> rr = tl::get_resource_reader (ex.get ());
if (! rr.first) {
throw tl::Exception (tl::to_string (tr ("Resource not found: ")) + abstract_path);
}
mp_delegate = rr.first;
needs_inflate = rr.second;
#endif
} else if (ex.test ("data:")) {
std::vector<unsigned char> data = tl::from_base64 (ex.get ());
char *data_ptr = new char [data.size ()];
memcpy (data_ptr, data.begin ().operator-> (), data.size ());
mp_delegate = new InputMemoryStream (data_ptr, data.size (), true);
} else if (ex.test ("pipe:")) {
mp_delegate = new InflatingInputPipe (ex.get ());
} else {
tl::URI uri (abstract_path);
if (uri.scheme () == "http" || uri.scheme () == "https") {
#if defined(HAVE_CURL) || defined(HAVE_QT)
mp_delegate = new InflatingInputHttpStream (abstract_path);
#else
throw tl::Exception (tl::to_string (tr ("HTTP support not enabled - HTTP/HTTPS paths are not available")));
#endif
} else if (uri.scheme () == "file") {
mp_delegate = new InputZLibFile (uri.path ());
} else if (! uri.scheme ().empty ()) {
throw tl::Exception (tl::to_string (tr ("URI scheme not supported: ")) + uri.scheme ());
} else {
mp_delegate = new InputZLibFile (abstract_path);
}
}
if (! mp_buffer) {
mp_buffer = new char [m_bcap];
}
m_owns_delegate = true;
if (needs_inflate) {
inflate_always ();
}
}
InputStream::~InputStream ()
{
if (mp_delegate && m_owns_delegate) {
delete mp_delegate;
mp_delegate = 0;
}
if (mp_inflate) {
delete mp_inflate;
mp_inflate = 0;
}
if (mp_buffer) {
delete[] mp_buffer;
mp_buffer = 0;
}
}
std::string InputStream::absolute_path (const std::string &abstract_path)
{
// TODO: align this implementation with InputStream ctor
tl::Extractor ex (abstract_path.c_str ());
#if defined(HAVE_QT)
if (ex.test (":")) {
return abstract_path;
} else
#endif
#if defined(HAVE_CURL) || defined(HAVE_QT)
if (ex.test ("http:") || ex.test ("https:")) {
return abstract_path;
} else
#endif
if (ex.test ("pipe:")) {
return abstract_path;
} else if (ex.test ("file:")) {
tl::URI uri (abstract_path);
return tl::absolute_path (uri.path ());
} else {
return tl::absolute_file_path (abstract_path);
}
}
const char *
InputStream::get (size_t n, bool bypass_inflate)
{
// if deflating, employ the deflate filter to get the data
if (mp_inflate && ! bypass_inflate) {
if (! mp_inflate->at_end ()) {
const char *r = mp_inflate->get (n);
tl_assert (r != 0); // since deflate did not report at_end()
return r;
} else if (m_stop_after_inflate) {
// report EOF after the inflator has finished
return 0;
} else {
delete mp_inflate;
mp_inflate = 0;
}
}
if (m_blen < n) {
// to keep move activity low, allocate twice as much as required
if (m_bcap < n * 2) {
while (m_bcap < n) {
m_bcap *= 2;
}
char *buffer = new char [m_bcap];
if (m_blen > 0) {
memcpy (buffer, mp_bptr, m_blen);
}
delete [] mp_buffer;
mp_buffer = buffer;
} else if (m_blen > 0) {
memmove (mp_buffer, mp_bptr, m_blen);
}
if (mp_delegate) {
m_blen += mp_delegate->read (mp_buffer + m_blen, m_bcap - m_blen);
}
mp_bptr = mp_buffer;
}
if (m_blen >= n) {
const char *r = mp_bptr;
mp_bptr += n;
m_blen -= n;
m_pos += n;
return r;
} else {
return 0;
}
}
void
InputStream::unget (size_t n)
{
if (n == 0) {
return;
}
if (mp_inflate) {
// TODO: this will not work if mp_inflate just got destroyed
// (no unget into previous compressed block)
mp_inflate->unget (n);
} else {
tl_assert (mp_buffer + n <= mp_bptr);
mp_bptr -= n;
m_blen += n;
m_pos -= n;
}
}
std::string
InputStream::read_all (size_t max_count)
{
std::string str;
if (mp_inflate) {
// Inflate is special - it does not have a guaranteed byte delivery, so we have to go the
// hard way and pick the file byte by byte
while (max_count > 0) {
const char *b = get (1);
if (b) {
str += *b;
--max_count;
} else {
break;
}
}
} else {
while (max_count > 0) {
size_t n = std::min (max_count, std::max (size_t (1), m_blen));
const char *b = get (n);
if (b) {
str += std::string (b, n);
max_count -= n;
} else {
break;
}
}
}
return str;
}
std::string
InputStream::read_all ()
{
std::string str;
if (mp_inflate) {
// Inflate is special - it does not have a guaranteed byte delivery, so we have to go the
// hard way and pick the file byte by byte
while (true) {
const char *b = get (1);
if (b) {
str += *b;
} else {
break;
}
}
} else {
while (true) {
size_t n = std::max (size_t (1), m_blen);
const char *b = get (n);
if (b) {
str += std::string (b, n);
} else {
break;
}
}
}
return str;
}
void InputStream::copy_to (tl::OutputStream &os)
{
const size_t chunk = 65536;
char b [chunk];
size_t read;
while (mp_delegate && (read = mp_delegate->read (b, sizeof (b))) > 0) {
os.put (b, read);
}
}
void
InputStream::inflate (bool stop_after)
{
tl_assert (mp_inflate == 0);
mp_inflate = new tl::InflateFilter (*this);
m_stop_after_inflate = stop_after;
}
void
InputStream::inflate_always ()
{
m_inflate_always = true;
reset ();
}
void
InputStream::close ()
{
if (mp_delegate) {
mp_delegate->close ();
}
}
void
InputStream::reset ()
{
// stop inflate
if (mp_inflate) {
delete mp_inflate;
mp_inflate = 0;
}
// optimize for a reset in the first m_bcap bytes
// -> this reduces the reset calls on mp_delegate which may not support this
if (m_pos < m_bcap) {
m_blen += m_pos;
mp_bptr = mp_buffer;
m_pos = 0;
} else {
tl_assert (mp_delegate != 0);
mp_delegate->reset ();
m_pos = 0;
if (mp_buffer) {
delete[] mp_buffer;
mp_buffer = 0;
}
mp_bptr = 0;
m_blen = 0;
mp_buffer = new char [m_bcap];
}
if (m_inflate_always) {
inflate ();
}
}
// ---------------------------------------------------------------
// TextInputStream implementation
TextInputStream::TextInputStream (InputStream &stream)
: m_line (1), m_next_line (1), m_at_end (false), m_stream (stream)
{
if (m_stream.get (1) == 0) {
m_at_end = true;
} else {
m_stream.unget (1);
}
}
std::string
TextInputStream::read_all ()
{
return read_all (std::numeric_limits<size_t>::max ());
}
std::string
TextInputStream::read_all (size_t max_count)
{
std::string text;
while (! at_end () && max_count > 0) {
char c = get_char ();
if (c == 0) {
break;
} else {
--max_count;
text += c;
}
}
return text;
}
const std::string &
TextInputStream::get_line ()
{
size_t line = m_next_line;
m_line_buffer.clear ();
while (! at_end ()) {
char c = get_char ();
if (c == '\n') {
// set at_end if there is nothing after this terminal LF -> this will avoid
// emitting an empty dummy line as the last one
if (peek_char () == 0) {
m_at_end = true;
}
break;
} else if (c == 0) {
break;
} else {
m_line_buffer += c;
}
}
m_line = line;
return m_line_buffer;
}
char
TextInputStream::get_char ()
{
while (true) {
m_line = m_next_line;
const char *c = m_stream.get (1);
if (c == 0) {
m_at_end = true;
return 0;
} else if (*c != '\r' && *c) {
if (*c == '\n') {
++m_next_line;
}
return *c;
}
}
}
char
TextInputStream::peek_char ()
{
while (true) {
m_line = m_next_line;
const char *c = m_stream.get (1);
if (c == 0) {
return 0;
} else if (*c != '\r' && *c) {
char cc = *c;
m_stream.unget (1);
return cc;
}
}
}
char
TextInputStream::skip ()
{
char c = 0;
while (! at_end () && isspace (c = peek_char ())) {
get_char ();
}
return at_end () ? 0 : c;
}
void
TextInputStream::reset ()
{
m_stream.reset ();
m_line = 1;
m_next_line = 1;
if (m_stream.get (1) == 0) {
m_at_end = true;
} else {
m_at_end = false;
m_stream.unget (1);
}
}
// ---------------------------------------------------------------
// InputFile implementation
InputFile::InputFile (const std::string &path)
: m_fd (-1)
{
m_source = tl::absolute_file_path (path);;
#if defined(_WIN32)
int fd = _wopen (tl::to_wstring (m_source).c_str (), _O_BINARY | _O_RDONLY | _O_SEQUENTIAL);
if (fd < 0) {
throw FileOpenErrorException (m_source, errno);
}
m_fd = fd;
#else
int fd = open (m_source.c_str (), O_RDONLY);
if (fd < 0) {
throw FileOpenErrorException (m_source, errno);
}
m_fd = fd;
#endif
}
InputFile::~InputFile ()
{
close ();
}
void
InputFile::close ()
{
if (m_fd >= 0) {
#if defined(_WIN32)
_close (m_fd);
#else
::close (m_fd);
#endif
m_fd = -1;
}
}
size_t
InputFile::read (char *b, size_t n)
{
tl_assert (m_fd >= 0);
#if defined(_WIN32)
ptrdiff_t ret = _read (m_fd, b, (unsigned int) n);
#else
ptrdiff_t ret = ::read (m_fd, b, (unsigned int) n);
#endif
if (ret < 0) {
throw FileReadErrorException (m_source, errno);
}
return size_t (ret);
}
void
InputFile::reset ()
{
if (m_fd >= 0) {
#if defined(_WIN64)
_lseeki64 (m_fd, 0, SEEK_SET);
#elif defined(_WIN64)
_lseek (m_fd, 0, SEEK_SET);
#else
lseek (m_fd, 0, SEEK_SET);
#endif
}
}
std::string
InputFile::absolute_path () const
{
return tl::absolute_file_path (m_source);
}
std::string
InputFile::filename () const
{
return tl::filename (m_source);
}
// ---------------------------------------------------------------
// InputZLibFile implementation
InputZLibFile::InputZLibFile (const std::string &path)
: mp_d (new ZLibFilePrivate ())
{
m_source = path;
std::string source = tl::absolute_file_path (path);
#if defined(_WIN32)
int fd = _wopen (tl::to_wstring (source).c_str (), _O_BINARY | _O_RDONLY | _O_SEQUENTIAL);
if (fd < 0) {
throw FileOpenErrorException (source, errno);
}
mp_d->zs = gzdopen (fd, "rb");
#else
mp_d->zs = gzopen (tl::string_to_system (source).c_str (), "rb");
#endif
if (mp_d->zs == NULL) {
throw FileOpenErrorException (source, errno);
}
}
InputZLibFile::~InputZLibFile ()
{
close ();
delete mp_d;
mp_d = 0;
}
void
InputZLibFile::close ()
{
if (mp_d->zs != NULL) {
gzclose (mp_d->zs);
mp_d->zs = NULL;
}
}
size_t
InputZLibFile::read (char *b, size_t n)
{
tl_assert (mp_d->zs != NULL);
int ret = gzread (mp_d->zs, b, (unsigned int) n);
if (ret < 0) {
int gz_err = 0;
const char *em = gzerror (mp_d->zs, &gz_err);
if (gz_err == Z_ERRNO) {
throw FileReadErrorException (m_source, errno);
} else {
throw ZLibReadErrorException (m_source, em);
}
}
return ret;
}
void
InputZLibFile::reset ()
{
if (mp_d->zs != NULL) {
gzrewind (mp_d->zs);
}
}
std::string
InputZLibFile::absolute_path () const
{
return tl::absolute_file_path (m_source);
}
std::string
InputZLibFile::filename () const
{
return tl::filename (m_source);
}
// ---------------------------------------------------------------
// OutputStream implementation
OutputStream::OutputStream (OutputStreamBase &delegate, bool as_text)
: m_pos (0), mp_delegate (&delegate), m_owns_delegate (false), m_as_text (as_text)
{
m_buffer_capacity = 16384;
m_buffer_pos = 0;
mp_buffer = new char[m_buffer_capacity];
}
OutputStream::OutputStream (OutputStreamBase *delegate, bool as_text)
: m_pos (0), mp_delegate (delegate), m_owns_delegate (true), m_as_text (as_text)
{
m_buffer_capacity = 16384;
m_buffer_pos = 0;
mp_buffer = new char[m_buffer_capacity];
}
OutputStream::OutputStreamMode
OutputStream::output_mode_from_filename (const std::string &abstract_path, OutputStream::OutputStreamMode om)
{
if (om == OM_Auto) {
if (tl::match_filename_to_format (abstract_path, "(*.gz *.gzip *.GZ *.GZIP)")) {
om = OM_Zlib;
} else {
om = OM_Plain;
}
}
return om;
}
static
OutputStreamBase *create_file_stream (const std::string &path, OutputStream::OutputStreamMode om, int keep_backups)
{
if (om == OutputStream::OM_Zlib) {
return new OutputZLibFile (path, keep_backups);
} else {
return new OutputFile (path, keep_backups);
}
}
OutputStream::OutputStream (const std::string &abstract_path, OutputStreamMode om, bool as_text, int keep_backups)
: m_pos (0), mp_delegate (0), m_owns_delegate (false), m_as_text (as_text), m_path (abstract_path)
{
// Determine output mode
om = output_mode_from_filename (abstract_path, om);
tl::Extractor ex (abstract_path.c_str ());
if (ex.test ("http:") || ex.test ("https:")) {
throw tl::Exception (tl::to_string (tr ("Cannot write to http:, https: or pipe: URL's")));
} else if (ex.test ("pipe:")) {
mp_delegate = new OutputPipe (ex.get ());
} else if (ex.test ("file:")) {
mp_delegate = create_file_stream (ex.get (), om, keep_backups);
} else {
mp_delegate = create_file_stream (abstract_path, om, keep_backups);
}
m_owns_delegate = true;
m_buffer_capacity = 16384;
m_buffer_pos = 0;
mp_buffer = new char[m_buffer_capacity];
}
OutputStream::~OutputStream ()
{
try {
close ();
} catch (...) {
// no recursive exceptions
}
}
void
OutputStream::close ()
{
try {
flush ();
if (mp_delegate && m_owns_delegate) {
delete mp_delegate;
mp_delegate = 0;
}
if (mp_buffer) {
delete[] mp_buffer;
mp_buffer = 0;
}
} catch (...) {
if (mp_delegate && m_owns_delegate) {
delete mp_delegate;
mp_delegate = 0;
}
if (mp_buffer) {
delete[] mp_buffer;
mp_buffer = 0;
}
throw;
}
}
void
OutputStream::set_as_text (bool f)
{
m_as_text = f;
}
inline void fast_copy (char *t, const char *s, size_t n)
{
if (n >= sizeof (unsigned long)) {
unsigned long *tl = reinterpret_cast<unsigned long *> (t);
const unsigned long *sl = reinterpret_cast<const unsigned long *> (s);
while (n >= sizeof (unsigned long)) {
*tl++ = *sl++;
n -= sizeof (unsigned long);
}
t = reinterpret_cast<char *> (tl);
s = reinterpret_cast<const char *> (sl);
}
while (n-- > 0) {
*t++ = *s++;
}
}
void
OutputStream::flush ()
{
if (m_buffer_pos > 0 && mp_delegate) {
mp_delegate->write (mp_buffer, m_buffer_pos);
m_buffer_pos = 0;
}
}
void
OutputStream::put (const char *b, size_t n)
{
if (! mp_delegate) {
return;
}
if (m_as_text) {
// skip CR, but replace LF by CRLF -> this will normalize the line terminators to CRLF
while (n > 0) {
if (*b == '\r') {
++b;
--n;
} else if (*b == '\n') {
const char *ls = line_separator ();
while (*ls) {
put_raw (ls++, 1);
}
++b;
--n;
} else {
const char *b0 = b;
while (n > 0 && *b != '\r' && *b != '\n') {
++b;
--n;
}
put_raw (b0, b - b0);
}
}
} else {
put_raw (b, n);
}
}
void
OutputStream::put_raw (const char *b, size_t n)
{
m_pos += n;
while (m_buffer_pos + n > m_buffer_capacity) {
size_t nw = m_buffer_capacity - m_buffer_pos;
if (nw) {
n -= nw;
fast_copy (mp_buffer + m_buffer_pos, b, nw);
b += nw;
}
mp_delegate->write (mp_buffer, m_buffer_capacity);
m_buffer_pos = 0;
}
if (n) {
fast_copy (mp_buffer + m_buffer_pos, b, n);
m_buffer_pos += n;
}
}
void
OutputStream::seek (size_t pos)
{
flush ();
if (mp_delegate) {
mp_delegate->seek (pos);
}
m_pos = pos;
}
// ---------------------------------------------------------------
// OutputFileBase implementation
OutputFileBase::OutputFileBase (const std::string &p, int keep_backups)
: m_keep_backups (keep_backups), m_path (tl::absolute_file_path (p)), m_has_error (false)
{
if (tl::file_exists (m_path)) {
m_backup_path = m_path + ".~backup";
if (tl::file_exists (m_backup_path)) {
if (! tl::rm_file (m_backup_path)) {
tl::warn << tl::sprintf (tl::to_string (tr ("Could not create backup file: unable to remove existing file '%s'")), m_backup_path);
m_backup_path = std::string ();
}
}
if (! m_backup_path.empty ()) {
if (! tl::rename_file (m_path, tl::filename (m_backup_path))) {
tl::warn << tl::sprintf (tl::to_string (tr ("Could not create backup file: unable to rename original file '%s' to backup file")), m_path, m_backup_path);
m_backup_path = std::string ();
}
}
}
}
OutputFileBase::~OutputFileBase ()
{
if (! m_backup_path.empty ()) {
if (m_has_error) {
if (! tl::rm_file (m_path)) {
tl::warn << tl::sprintf (tl::to_string (tr ("Could not restore backup file: unable to remove file '%s'")), m_path);
} else if (! tl::rename_file (m_backup_path, m_path)) {
tl::warn << tl::sprintf (tl::to_string (tr ("Could not restore backup file: unable to rename file '%s' back to '%s'")), m_backup_path, m_path);
}
} else {
if (m_keep_backups == 0) {
if (! tl::rm_file (m_backup_path)) {
tl::warn << tl::sprintf (tl::to_string (tr ("Could not remove backup file '%s'")), m_backup_path);
}
} else {
// shuffle backup files
int n = 1;
for ( ; m_keep_backups < 0 || n < m_keep_backups; ++n) {
std::string p = m_path + "." + tl::to_string (n);
if (! tl::file_exists (p)) {
break;
}
}
while (n > 0) {
std::string p = m_path + "." + tl::to_string (n);
std::string pprev = n > 1 ? (m_path + "." + tl::to_string (n - 1)) : m_backup_path;
if (tl::file_exists (p)) {
if (! tl::rm_file (p)) {
tl::warn << tl::sprintf (tl::to_string (tr ("Error shuffling backup files: unable to remove file '%s'")), p);
}
}
if (! tl::rename_file (pprev, p)) {
tl::warn << tl::sprintf (tl::to_string (tr ("Error shuffling backup files: unable to rename file '%s' to '%s'")), pprev, p);
}
--n;
}
}
}
}
}
void OutputFileBase::seek (size_t s)
{
try {
seek_file (s);
} catch (...) {
reject ();
throw;
}
}
void OutputFileBase::write (const char *b, size_t n)
{
try {
write_file (b, n);
} catch (...) {
reject ();
throw;
}
}
void OutputFileBase::reject ()
{
m_has_error = true;
}
// ---------------------------------------------------------------
// OutputFile implementation
OutputFile::OutputFile (const std::string &p, int keep_backups)
: OutputFileBase (p, keep_backups), m_fd (-1)
{
#if defined(_WIN32)
int fd = _wopen (tl::to_wstring (path ()).c_str (), _O_CREAT | _O_TRUNC | _O_BINARY | _O_WRONLY | _O_SEQUENTIAL, _S_IREAD | _S_IWRITE );
if (fd < 0) {
throw FileOpenErrorException (path (), errno);
}
m_fd = fd;
#else
int fd = open (path ().c_str (), O_WRONLY | O_CREAT | O_TRUNC, 0666);
if (fd < 0) {
throw FileOpenErrorException (path (), errno);
}
m_fd = fd;
#endif
}
OutputFile::~OutputFile ()
{
if (m_fd >= 0) {
#if defined(_WIN32)
_close (m_fd);
#else
close (m_fd);
#endif
m_fd = -1;
}
}
void
OutputFile::seek_file (size_t s)
{
tl_assert (m_fd >= 0);
#if defined(_WIN64)
_lseeki64 (m_fd, s, SEEK_SET);
#elif defined(_WIN32)
_lseek (m_fd, s, SEEK_SET);
#else
lseek (m_fd, s, SEEK_SET);
#endif
}
void
OutputFile::write_file (const char *b, size_t n)
{
tl_assert (m_fd >= 0);
#if defined(_WIN32)
ptrdiff_t ret = _write (m_fd, b, (unsigned int) n);
#else
ptrdiff_t ret = ::write (m_fd, b, (unsigned int) n);
#endif
if (ret < 0) {
throw FileWriteErrorException (path (), errno);
}
}
// ---------------------------------------------------------------
// OutputZLibFile implementation
OutputZLibFile::OutputZLibFile (const std::string &p, int keep_backups)
: OutputFileBase (p, keep_backups), mp_d (new ZLibFilePrivate ())
{
#if defined(_WIN32)
FILE *file = _wfopen (tl::to_wstring (path ()).c_str (), L"wb");
if (file == NULL) {
throw FileOpenErrorException (path (), errno);
}
mp_d->zs = gzdopen (_fileno (file), "wb");
#else
mp_d->zs = gzopen (tl::string_to_system (path ()).c_str (), "wb");
#endif
if (mp_d->zs == NULL) {
throw FileOpenErrorException (path (), errno);
}
}
OutputZLibFile::~OutputZLibFile ()
{
if (mp_d->zs != NULL) {
gzclose (mp_d->zs);
mp_d->zs = NULL;
}
delete mp_d;
mp_d = 0;
}
void
OutputZLibFile::write_file (const char *b, size_t n)
{
tl_assert (mp_d->zs != NULL);
int ret = gzwrite (mp_d->zs, (char *) b, (unsigned int) n);
if (ret < 0) {
int gz_err = 0;
const char *em = gzerror (mp_d->zs, &gz_err);
if (gz_err == Z_ERRNO) {
throw FileWriteErrorException (path (), errno);
} else {
throw ZLibWriteErrorException (path (), em);
}
}
}
#if defined(_WIN32)
// ---------------------------------------------------------------
// InputPipe delegate implementation
InputPipe::InputPipe (const std::string &path)
: m_file (NULL)
{
std::wstring wpath = tl::to_wstring (path);
m_source = path;
m_file = _wpopen (wpath.c_str (), L"r");
if (m_file == NULL) {
throw FilePOpenErrorException (m_source, errno);
}
}
InputPipe::~InputPipe ()
{
close ();
}
void
InputPipe::close ()
{
wait ();
}
int InputPipe::wait ()
{
int ret = 0;
if (m_file != NULL) {
ret = _pclose (m_file);
m_file = NULL;
}
return ret;
}
size_t
InputPipe::read (char *b, size_t n)
{
tl_assert (m_file != NULL);
size_t ret = fread (b, 1, n, m_file);
if (ret < n) {
if (ferror (m_file)) {
throw FilePReadErrorException (m_source, errno);
}
}
return ret;
}
void
InputPipe::reset ()
{
throw tl::Exception (tl::to_string (tr ("'reset' is not supported on pipeline input files")));
}
// ---------------------------------------------------------------
// OutputPipe delegate implementation
OutputPipe::OutputPipe (const std::string &path)
: m_file (NULL)
{
std::wstring wpath = tl::to_wstring (path);
m_source = path;
m_file = _wpopen (wpath.c_str (), L"w");
if (m_file == NULL) {
throw FilePOpenErrorException (m_source, errno);
}
}
OutputPipe::~OutputPipe ()
{
if (m_file != NULL) {
_pclose (m_file);
m_file = NULL;
}
}
void
OutputPipe::write (const char *b, size_t n)
{
tl_assert (m_file != NULL);
size_t ret = fwrite (b, 1, n, m_file);
if (ret < n) {
if (ferror (m_file)) {
throw FilePWriteErrorException (m_source, errno);
}
}
}
#else
// ---------------------------------------------------------------
// InputPipe delegate implementation
InputPipe::InputPipe (const std::string &source)
: m_file (NULL)
{
m_source = source;
m_file = popen (tl::string_to_system (source).c_str (), "r");
if (m_file == NULL) {
throw FilePOpenErrorException (m_source, errno);
}
}
InputPipe::~InputPipe ()
{
close ();
}
void
InputPipe::close ()
{
wait ();
}
int InputPipe::wait ()
{
int ret = 0;
if (m_file != NULL) {
ret = pclose (m_file);
m_file = NULL;
}
return ret;
}
size_t
InputPipe::read (char *b, size_t n)
{
tl_assert (m_file != NULL);
bool retry = true;
size_t ret = 0;
while (retry) {
retry = false;
ret = fread (b, 1, n, m_file);
if (ret < n) {
if (ferror (m_file)) {
if (errno != EINTR) {
throw FilePReadErrorException (m_source, errno);
} else if (ret == 0) {
retry = true;
clearerr (m_file);
}
}
}
}
return ret;
}
void
InputPipe::reset ()
{
throw tl::Exception (tl::to_string (tr ("'reset' is not supported on pipeline input files")));
}
// ---------------------------------------------------------------
// OutputPipe delegate implementation
OutputPipe::OutputPipe (const std::string &path)
: m_file (NULL)
{
m_source = path;
m_file = popen (tl::string_to_system (path).c_str (), "w");
if (m_file == NULL) {
throw FilePOpenErrorException (m_source, errno);
}
}
OutputPipe::~OutputPipe ()
{
if (m_file != NULL) {
pclose (m_file);
m_file = NULL;
}
}
void
OutputPipe::write (const char *b, size_t n)
{
tl_assert (m_file != NULL);
size_t ret = fwrite (b, 1, n, m_file);
if (ret < n) {
if (ferror (m_file) && errno != EINTR) {
throw FilePReadErrorException (m_source, errno);
}
}
}
#endif
// ---------------------------------------------------------------
// match_filename_to_format implementation
bool
match_filename_to_format (const std::string &fn, const std::string &fmt)
{
const char *fp = fmt.c_str ();
while (*fp && *fp != '(') {
++fp;
}
while (*fp && *fp != ')') {
if (*++fp == '*') {
++fp;
}
const char *fpp = fp;
while (*fpp && *fpp != ' ' && *fpp != ')') {
++fpp;
}
if (fn.size () > (unsigned int) (fpp - fp) && strncmp (fn.c_str () + fn.size () - (fpp - fp), fp, fpp - fp) == 0) {
return true;
}
fp = fpp;
while (*fp == ' ') {
++fp;
}
}
return false;
}
}