From 8c744efd5693f2a39b1441d747ebdf0d23faa308 Mon Sep 17 00:00:00 2001 From: Max Kellermann Date: Fri, 9 Sep 2016 18:47:42 +0200 Subject: [PATCH] input/InputStream: migrate from class Error to C++ exceptions --- src/TagFile.cxx | 8 +++- src/TagStream.cxx | 7 ++- src/archive/plugins/Bzip2ArchivePlugin.cxx | 24 ++++------ src/archive/plugins/Iso9660ArchivePlugin.cxx | 18 +++----- src/archive/plugins/ZzipArchivePlugin.cxx | 27 ++++------- src/decoder/DecoderAPI.cxx | 11 +---- src/decoder/DecoderThread.cxx | 9 ++-- .../plugins/AudiofileDecoderPlugin.cxx | 10 +++-- src/decoder/plugins/DsdLib.cxx | 27 +++++++---- src/decoder/plugins/FaadDecoderPlugin.cxx | 7 ++- src/decoder/plugins/FfmpegDecoderPlugin.cxx | 8 +++- src/decoder/plugins/FfmpegIo.cxx | 11 +++-- src/decoder/plugins/FlacDecoderPlugin.cxx | 5 ++- src/decoder/plugins/FlacIOHandle.cxx | 42 +++++++++-------- src/decoder/plugins/FlacInput.cxx | 13 +++--- src/decoder/plugins/MadDecoderPlugin.cxx | 8 +++- src/decoder/plugins/MpcdecDecoderPlugin.cxx | 9 +++- src/decoder/plugins/OggDecoder.cxx | 17 +++---- src/decoder/plugins/OggDecoder.hxx | 4 +- src/decoder/plugins/OpusDecoderPlugin.cxx | 12 ++++- src/decoder/plugins/PcmDecoderPlugin.cxx | 10 +++-- src/decoder/plugins/SndfileDecoderPlugin.cxx | 12 ++--- src/decoder/plugins/VorbisDecoderPlugin.cxx | 15 ++++--- src/decoder/plugins/WavpackDecoderPlugin.cxx | 14 +++++- src/input/AsyncInputStream.cxx | 24 ++++------ src/input/AsyncInputStream.hxx | 6 +-- src/input/IcyInputStream.cxx | 6 +-- src/input/IcyInputStream.hxx | 2 +- src/input/InputStream.cxx | 45 +++++++++---------- src/input/InputStream.hxx | 41 ++++++++++------- src/input/Open.cxx | 11 +---- src/input/ProxyInputStream.cxx | 17 ++++--- src/input/ProxyInputStream.hxx | 6 +-- src/input/Reader.cxx | 11 +---- src/input/TextInputStream.cxx | 15 ++++--- src/input/ThreadInputStream.cxx | 8 ++-- src/input/ThreadInputStream.hxx | 4 +- src/input/plugins/CdioParanoiaInputPlugin.cxx | 30 +++++-------- src/input/plugins/FfmpegInputPlugin.cxx | 21 ++++----- src/input/plugins/FileInputPlugin.cxx | 21 +++------ src/input/plugins/RewindInputPlugin.cxx | 17 +++---- src/input/plugins/SmbclientInputPlugin.cxx | 24 ++++------ src/lib/expat/ExpatParser.cxx | 10 ++--- src/lib/expat/ExpatParser.hxx | 6 +-- src/lib/ffmpeg/Error.cxx | 9 ++++ src/lib/ffmpeg/Error.hxx | 3 ++ src/lib/xiph/OggFind.cxx | 12 +++-- src/playlist/PlaylistRegistry.cxx | 13 ++++-- src/playlist/plugins/AsxPlaylistPlugin.cxx | 8 +--- src/playlist/plugins/RssPlaylistPlugin.cxx | 8 +--- .../plugins/SoundCloudPlaylistPlugin.cxx | 18 ++------ src/playlist/plugins/XspfPlaylistPlugin.cxx | 8 +--- src/tag/Aiff.cxx | 24 ++++------ src/tag/Aiff.hxx | 5 ++- src/tag/ApeLoader.cxx | 15 ++++--- src/tag/Generic.cxx | 5 ++- src/tag/Id3Load.cxx | 43 ++++++------------ src/tag/Riff.cxx | 24 ++++------ src/tag/Riff.hxx | 5 ++- test/DumpDatabase.cxx | 3 +- test/FakeDecoderAPI.cxx | 9 +++- test/dump_text_file.cxx | 8 +--- test/run_input.cxx | 17 ++----- test/test_rewind.cxx | 23 +++++----- 64 files changed, 440 insertions(+), 473 deletions(-) diff --git a/src/TagFile.cxx b/src/TagFile.cxx index 8c6ccb61a..5c965832b 100644 --- a/src/TagFile.cxx +++ b/src/TagFile.cxx @@ -68,8 +68,12 @@ public: } catch (const std::runtime_error &) { return false; } - } else - is->LockRewind(IgnoreError()); + } else { + try { + is->LockRewind(); + } catch (const std::runtime_error &) { + } + } /* now try the stream_tag() method */ return plugin.ScanStream(*is, handler, handler_ctx); diff --git a/src/TagStream.cxx b/src/TagStream.cxx index 45b18e60f..78bed0513 100644 --- a/src/TagStream.cxx +++ b/src/TagStream.cxx @@ -31,6 +31,8 @@ #include "thread/Mutex.hxx" #include "thread/Cond.hxx" +#include + #include /** @@ -63,7 +65,10 @@ tag_stream_scan(InputStream &is, const TagHandler &handler, void *ctx) return decoder_plugins_try([suffix, mime, &is, &handler, ctx](const DecoderPlugin &plugin){ - is.LockRewind(IgnoreError()); + try { + is.LockRewind(); + } catch (const std::runtime_error &) { + } return CheckDecoderPlugin(plugin, suffix, mime) && plugin.ScanStream(is, handler, ctx); diff --git a/src/archive/plugins/Bzip2ArchivePlugin.cxx b/src/archive/plugins/Bzip2ArchivePlugin.cxx index 89ccc0a93..859a4de4f 100644 --- a/src/archive/plugins/Bzip2ArchivePlugin.cxx +++ b/src/archive/plugins/Bzip2ArchivePlugin.cxx @@ -30,8 +30,6 @@ #include "input/LocalOpen.hxx" #include "thread/Cond.hxx" #include "util/RefCount.hxx" -#include "util/Error.hxx" -#include "util/Domain.hxx" #include "fs/Path.hxx" #include @@ -96,15 +94,13 @@ public: /* virtual methods from InputStream */ bool IsEOF() override; - size_t Read(void *ptr, size_t size, Error &error) override; + size_t Read(void *ptr, size_t size) override; private: void Open(); - bool FillBuffer(Error &error); + bool FillBuffer(); }; -static constexpr Domain bz2_domain("bz2"); - /* single archive handling allocation helpers */ inline void @@ -161,13 +157,12 @@ Bzip2ArchiveFile::OpenStream(const char *path, } inline bool -Bzip2InputStream::FillBuffer(Error &error) +Bzip2InputStream::FillBuffer() { if (bzstream.avail_in > 0) return true; - size_t count = archive->istream->Read(buffer, sizeof(buffer), - error); + size_t count = archive->istream->Read(buffer, sizeof(buffer)); if (count == 0) return false; @@ -177,7 +172,7 @@ Bzip2InputStream::FillBuffer(Error &error) } size_t -Bzip2InputStream::Read(void *ptr, size_t length, Error &error) +Bzip2InputStream::Read(void *ptr, size_t length) { int bz_result; size_t nbytes = 0; @@ -189,7 +184,7 @@ Bzip2InputStream::Read(void *ptr, size_t length, Error &error) bzstream.avail_out = length; do { - if (!FillBuffer(error)) + if (!FillBuffer()) return 0; bz_result = BZ2_bzDecompress(&bzstream); @@ -199,11 +194,8 @@ Bzip2InputStream::Read(void *ptr, size_t length, Error &error) break; } - if (bz_result != BZ_OK) { - error.Set(bz2_domain, bz_result, - "BZ2_bzDecompress() has failed"); - return 0; - } + if (bz_result != BZ_OK) + throw std::runtime_error("BZ2_bzDecompress() has failed"); } while (bzstream.avail_out == length); nbytes = length - bzstream.avail_out; diff --git a/src/archive/plugins/Iso9660ArchivePlugin.cxx b/src/archive/plugins/Iso9660ArchivePlugin.cxx index 95f0b9830..225b0db4a 100644 --- a/src/archive/plugins/Iso9660ArchivePlugin.cxx +++ b/src/archive/plugins/Iso9660ArchivePlugin.cxx @@ -30,8 +30,6 @@ #include "fs/Path.hxx" #include "util/RefCount.hxx" #include "util/RuntimeError.hxx" -#include "util/Error.hxx" -#include "util/Domain.hxx" #include @@ -82,8 +80,6 @@ public: Mutex &mutex, Cond &cond) override; }; -static constexpr Domain iso9660_domain("iso9660"); - /* archive open && listing routine */ inline void @@ -167,7 +163,7 @@ public: /* virtual methods from InputStream */ bool IsEOF() override; - size_t Read(void *ptr, size_t size, Error &error) override; + size_t Read(void *ptr, size_t size) override; }; InputStream * @@ -184,7 +180,7 @@ Iso9660ArchiveFile::OpenStream(const char *pathname, } size_t -Iso9660InputStream::Read(void *ptr, size_t read_size, Error &error) +Iso9660InputStream::Read(void *ptr, size_t read_size) { int readed = 0; int no_blocks, cur_block; @@ -204,12 +200,10 @@ Iso9660InputStream::Read(void *ptr, size_t read_size, Error &error) readed = archive.SeekRead(ptr, statbuf->lsn + cur_block, no_blocks); - if (readed != no_blocks * ISO_BLOCKSIZE) { - error.Format(iso9660_domain, - "error reading ISO file at lsn %lu", - (unsigned long)cur_block); - return 0; - } + if (readed != no_blocks * ISO_BLOCKSIZE) + throw FormatRuntimeError("error reading ISO file at lsn %lu", + (unsigned long)cur_block); + if (left_bytes < read_size) { readed = left_bytes; } diff --git a/src/archive/plugins/ZzipArchivePlugin.cxx b/src/archive/plugins/ZzipArchivePlugin.cxx index 506a29703..5bf64eed1 100644 --- a/src/archive/plugins/ZzipArchivePlugin.cxx +++ b/src/archive/plugins/ZzipArchivePlugin.cxx @@ -30,8 +30,6 @@ #include "fs/Path.hxx" #include "util/RefCount.hxx" #include "util/RuntimeError.hxx" -#include "util/Error.hxx" -#include "util/Domain.hxx" #include @@ -63,8 +61,6 @@ public: Mutex &mutex, Cond &cond) override; }; -static constexpr Domain zzip_domain("zzip"); - /* archive open && listing routine */ static ArchiveFile * @@ -121,8 +117,8 @@ struct ZzipInputStream final : public InputStream { /* virtual methods from InputStream */ bool IsEOF() override; - size_t Read(void *ptr, size_t size, Error &error) override; - bool Seek(offset_type offset, Error &error) override; + size_t Read(void *ptr, size_t size) override; + void Seek(offset_type offset) override; }; InputStream * @@ -140,13 +136,11 @@ ZzipArchiveFile::OpenStream(const char *pathname, } size_t -ZzipInputStream::Read(void *ptr, size_t read_size, Error &error) +ZzipInputStream::Read(void *ptr, size_t read_size) { int ret = zzip_file_read(file, ptr, read_size); - if (ret < 0) { - error.Set(zzip_domain, "zzip_file_read() has failed"); - return 0; - } + if (ret < 0) + throw std::runtime_error("zzip_file_read() has failed"); offset = zzip_tell(file); return ret; @@ -158,17 +152,14 @@ ZzipInputStream::IsEOF() return offset_type(zzip_tell(file)) == size; } -bool -ZzipInputStream::Seek(offset_type new_offset, Error &error) +void +ZzipInputStream::Seek(offset_type new_offset) { zzip_off_t ofs = zzip_seek(file, new_offset, SEEK_SET); - if (ofs < 0) { - error.Set(zzip_domain, "zzip_seek() has failed"); - return false; - } + if (ofs < 0) + throw std::runtime_error("zzip_seek() has failed"); offset = ofs; - return true; } /* exported structures */ diff --git a/src/decoder/DecoderAPI.cxx b/src/decoder/DecoderAPI.cxx index 2dda85063..3758a922e 100644 --- a/src/decoder/DecoderAPI.cxx +++ b/src/decoder/DecoderAPI.cxx @@ -334,15 +334,8 @@ try { is.cond.wait(is.mutex); } - Error error; - size_t nbytes = is.Read(buffer, length, error); - assert(nbytes == 0 || !error.IsDefined()); - assert(nbytes > 0 || error.IsDefined() || is.IsEOF()); - - lock.Unlock(); - - if (gcc_unlikely(nbytes == 0 && error.IsDefined())) - LogError(error); + size_t nbytes = is.Read(buffer, length); + assert(nbytes > 0 || is.IsEOF()); return nbytes; } catch (const std::runtime_error &e) { diff --git a/src/decoder/DecoderThread.cxx b/src/decoder/DecoderThread.cxx index 71634724f..1e771940b 100644 --- a/src/decoder/DecoderThread.cxx +++ b/src/decoder/DecoderThread.cxx @@ -73,9 +73,7 @@ decoder_input_stream_open(DecoderControl &dc, const char *uri) is->Update(); } - Error error; - if (!is->Check(error)) - throw error; + is->Check(); return is; } @@ -112,7 +110,10 @@ decoder_stream_decode(const DecoderPlugin &plugin, throw StopDecoder(); /* rewind the stream, so each plugin gets a fresh start */ - input_stream.Rewind(IgnoreError()); + try { + input_stream.Rewind(); + } catch (const std::runtime_error &) { + } { const ScopeUnlock unlock(decoder.dc.mutex); diff --git a/src/decoder/plugins/AudiofileDecoderPlugin.cxx b/src/decoder/plugins/AudiofileDecoderPlugin.cxx index f26d6f880..17abdf1b6 100644 --- a/src/decoder/plugins/AudiofileDecoderPlugin.cxx +++ b/src/decoder/plugins/AudiofileDecoderPlugin.cxx @@ -30,6 +30,8 @@ #include #include +#include + #include #include @@ -115,11 +117,11 @@ audiofile_file_seek(AFvirtualfile *vfile, AFfileoffset _offset, if (is_relative) offset += is.GetOffset(); - Error error; - if (is.LockSeek(offset, error)) { + try { + is.LockSeek(offset); return is.GetOffset(); - } else { - LogError(error, "Seek failed"); + } catch (const std::runtime_error &e) { + LogError(e, "Seek failed"); return -1; } } diff --git a/src/decoder/plugins/DsdLib.cxx b/src/decoder/plugins/DsdLib.cxx index a5e9c8391..44ea7fc4b 100644 --- a/src/decoder/plugins/DsdLib.cxx +++ b/src/decoder/plugins/DsdLib.cxx @@ -28,15 +28,16 @@ #include "../DecoderAPI.hxx" #include "input/InputStream.hxx" #include "tag/TagId3.hxx" -#include "util/Error.hxx" - -#include -#include #ifdef ENABLE_ID3TAG #include #endif +#include + +#include +#include + bool DsdId::Equals(const char *s) const { @@ -53,8 +54,13 @@ bool dsdlib_skip_to(Decoder *decoder, InputStream &is, offset_type offset) { - if (is.IsSeekable()) - return is.LockSeek(offset, IgnoreError()); + if (is.IsSeekable()) { + try { + is.LockSeek(offset); + } catch (const std::runtime_error &) { + return false; + } + } if (is.GetOffset() > offset) return false; @@ -72,8 +78,13 @@ dsdlib_skip(Decoder *decoder, InputStream &is, if (delta == 0) return true; - if (is.IsSeekable()) - return is.LockSeek(is.GetOffset() + delta, IgnoreError()); + if (is.IsSeekable()) { + try { + is.LockSeek(is.GetOffset() + delta); + } catch (const std::runtime_error &) { + return false; + } + } if (delta > 1024 * 1024) /* don't skip more than one megabyte; it would be too diff --git a/src/decoder/plugins/FaadDecoderPlugin.cxx b/src/decoder/plugins/FaadDecoderPlugin.cxx index 04bb000ee..ad1470fa6 100644 --- a/src/decoder/plugins/FaadDecoderPlugin.cxx +++ b/src/decoder/plugins/FaadDecoderPlugin.cxx @@ -31,6 +31,8 @@ #include +#include + #include #include @@ -192,7 +194,10 @@ faad_song_duration(DecoderBuffer &buffer, InputStream &is) auto song_length = adts_song_duration(buffer); - is.LockSeek(tagsize, IgnoreError()); + try { + is.LockSeek(tagsize); + } catch (const std::runtime_error &) { + } buffer.Clear(); diff --git a/src/decoder/plugins/FfmpegDecoderPlugin.cxx b/src/decoder/plugins/FfmpegDecoderPlugin.cxx index 662ba2af9..76e595041 100644 --- a/src/decoder/plugins/FfmpegDecoderPlugin.cxx +++ b/src/decoder/plugins/FfmpegDecoderPlugin.cxx @@ -460,9 +460,15 @@ ffmpeg_probe(Decoder *decoder, InputStream &is) unsigned char buffer[BUFFER_SIZE]; size_t nbytes = decoder_read(decoder, is, buffer, BUFFER_SIZE); - if (nbytes <= PADDING || !is.LockRewind(IgnoreError())) + if (nbytes <= PADDING) return nullptr; + try { + is.LockRewind(); + } catch (const std::runtime_error &) { + return nullptr; + } + /* some ffmpeg parsers (e.g. ac3_parser.c) read a few bytes beyond the declared buffer limit, which makes valgrind angry; this workaround removes some padding from the buffer diff --git a/src/decoder/plugins/FfmpegIo.cxx b/src/decoder/plugins/FfmpegIo.cxx index cf50a8013..b8c6f918f 100644 --- a/src/decoder/plugins/FfmpegIo.cxx +++ b/src/decoder/plugins/FfmpegIo.cxx @@ -24,7 +24,8 @@ #include "FfmpegIo.hxx" #include "../DecoderAPI.hxx" #include "input/InputStream.hxx" -#include "util/Error.hxx" + +#include AvioStream::~AvioStream() { @@ -68,10 +69,12 @@ AvioStream::Seek(int64_t pos, int whence) return -1; } - if (!input.LockSeek(pos, IgnoreError())) + try { + input.LockSeek(pos); + return input.GetOffset(); + } catch (const std::runtime_error &) { return -1; - - return input.GetOffset(); + } } int diff --git a/src/decoder/plugins/FlacDecoderPlugin.cxx b/src/decoder/plugins/FlacDecoderPlugin.cxx index 5b70ba12b..2b9d7bb72 100644 --- a/src/decoder/plugins/FlacDecoderPlugin.cxx +++ b/src/decoder/plugins/FlacDecoderPlugin.cxx @@ -330,7 +330,10 @@ oggflac_decode(Decoder &decoder, InputStream &input_stream) /* rewind the stream, because ogg_codec_detect() has moved it */ - input_stream.LockRewind(IgnoreError()); + try { + input_stream.LockRewind(); + } catch (const std::runtime_error &) { + } flac_decode_internal(decoder, input_stream, true); } diff --git a/src/decoder/plugins/FlacIOHandle.cxx b/src/decoder/plugins/FlacIOHandle.cxx index 84f25ab48..d3bcd630e 100644 --- a/src/decoder/plugins/FlacIOHandle.cxx +++ b/src/decoder/plugins/FlacIOHandle.cxx @@ -19,10 +19,11 @@ #include "config.h" #include "FlacIOHandle.hxx" -#include "util/Error.hxx" #include "Log.hxx" #include "Compiler.h" +#include + #include #include @@ -37,24 +38,29 @@ FlacIORead(void *ptr, size_t size, size_t nmemb, FLAC__IOHandle handle) /* libFLAC is very picky about short reads, and expects the IO callback to fill the whole buffer (undocumented!) */ - Error error; while (p < end) { - size_t nbytes = is->LockRead(p, end - p, error); - if (nbytes == 0) { - if (!error.IsDefined()) + try { + size_t nbytes = is->LockRead(p, end - p); + if (nbytes == 0) /* end of file */ break; - if (error.IsDomain(errno_domain)) - errno = error.GetCode(); - else - /* just some random non-zero - errno value */ - errno = EINVAL; + p += nbytes; + +#ifndef WIN32 + } catch (const std::system_error &e) { + errno = e.code().category() == std::system_category() + ? e.code().value() + /* just some random non-zero errno + value */ + : EINVAL; + return 0; +#endif + } catch (const std::runtime_error &) { + /* just some random non-zero errno value */ + errno = EINVAL; return 0; } - - p += nbytes; } /* libFLAC expects a clean errno after returning from the IO @@ -88,13 +94,13 @@ FlacIOSeek(FLAC__IOHandle handle, FLAC__int64 _offset, int whence) return -1; } - Error error; - if (!is->LockSeek(offset, error)) { - LogError(error); + try { + is->LockSeek(offset); + return 0; + } catch (const std::runtime_error &e) { + LogError(e); return -1; } - - return 0; } static FLAC__int64 diff --git a/src/decoder/plugins/FlacInput.cxx b/src/decoder/plugins/FlacInput.cxx index 063eeae54..016c5beb2 100644 --- a/src/decoder/plugins/FlacInput.cxx +++ b/src/decoder/plugins/FlacInput.cxx @@ -22,10 +22,11 @@ #include "FlacDomain.hxx" #include "../DecoderAPI.hxx" #include "input/InputStream.hxx" -#include "util/Error.hxx" #include "Log.hxx" #include "Compiler.h" +#include + FLAC__StreamDecoderReadStatus FlacInput::Read(FLAC__byte buffer[], size_t *bytes) { @@ -50,13 +51,13 @@ FlacInput::Seek(FLAC__uint64 absolute_byte_offset) if (!input_stream.IsSeekable()) return FLAC__STREAM_DECODER_SEEK_STATUS_UNSUPPORTED; - ::Error error; - if (!input_stream.LockSeek(absolute_byte_offset, error)) { - LogError(error); + try { + input_stream.LockSeek(absolute_byte_offset); + return FLAC__STREAM_DECODER_SEEK_STATUS_OK; + } catch (const std::runtime_error &e) { + LogError(e); return FLAC__STREAM_DECODER_SEEK_STATUS_ERROR; } - - return FLAC__STREAM_DECODER_SEEK_STATUS_OK; } FLAC__StreamDecoderTellStatus diff --git a/src/decoder/plugins/MadDecoderPlugin.cxx b/src/decoder/plugins/MadDecoderPlugin.cxx index cbcd72d6e..35972b046 100644 --- a/src/decoder/plugins/MadDecoderPlugin.cxx +++ b/src/decoder/plugins/MadDecoderPlugin.cxx @@ -39,6 +39,8 @@ #include #endif +#include + #include #include #include @@ -217,9 +219,11 @@ MadDecoder::MadDecoder(Decoder *_decoder, inline bool MadDecoder::Seek(long offset) { - Error error; - if (!input_stream.LockSeek(offset, error)) + try { + input_stream.LockSeek(offset); + } catch (const std::runtime_error &) { return false; + } mad_stream_buffer(&stream, input_buffer, 0); stream.error = MAD_ERROR_NONE; diff --git a/src/decoder/plugins/MpcdecDecoderPlugin.cxx b/src/decoder/plugins/MpcdecDecoderPlugin.cxx index 7b650fede..ee14fcea2 100644 --- a/src/decoder/plugins/MpcdecDecoderPlugin.cxx +++ b/src/decoder/plugins/MpcdecDecoderPlugin.cxx @@ -32,6 +32,8 @@ #include +#include + #include struct mpc_decoder_data { @@ -62,7 +64,12 @@ mpc_seek_cb(mpc_reader *reader, mpc_int32_t offset) struct mpc_decoder_data *data = (struct mpc_decoder_data *)reader->data; - return data->is.LockSeek(offset, IgnoreError()); + try { + data->is.LockSeek(offset); + return true; + } catch (const std::runtime_error &) { + return false; + } } static mpc_int32_t diff --git a/src/decoder/plugins/OggDecoder.cxx b/src/decoder/plugins/OggDecoder.cxx index b987c45b0..b70e8f8f7 100644 --- a/src/decoder/plugins/OggDecoder.cxx +++ b/src/decoder/plugins/OggDecoder.cxx @@ -21,7 +21,8 @@ #include "OggDecoder.hxx" #include "lib/xiph/OggFind.hxx" #include "input/InputStream.hxx" -#include "util/Error.hxx" + +#include /** * Load the end-of-stream packet and restore the previous file @@ -52,7 +53,10 @@ OggDecoder::LoadEndPacket(ogg_packet &packet) const } /* restore the previous file position */ - input_stream.LockSeek(old_offset, IgnoreError()); + try { + input_stream.LockSeek(old_offset); + } catch (const std::runtime_error &) { + } return result; } @@ -67,8 +71,8 @@ OggDecoder::LoadEndGranulePos() const return packet.granulepos; } -bool -OggDecoder::SeekGranulePos(ogg_int64_t where_granulepos, Error &error) +void +OggDecoder::SeekGranulePos(ogg_int64_t where_granulepos) { assert(IsSeekable()); @@ -78,10 +82,7 @@ OggDecoder::SeekGranulePos(ogg_int64_t where_granulepos, Error &error) offset_type offset(where_granulepos * input_stream.GetSize() / end_granulepos); - if (!input_stream.LockSeek(offset, error)) - return false; - + input_stream.LockSeek(offset); PostSeek(); - return true; } diff --git a/src/decoder/plugins/OggDecoder.hxx b/src/decoder/plugins/OggDecoder.hxx index bb33f69cc..7534653ab 100644 --- a/src/decoder/plugins/OggDecoder.hxx +++ b/src/decoder/plugins/OggDecoder.hxx @@ -24,8 +24,6 @@ #include "lib/xiph/OggVisitor.hxx" #include "decoder/Reader.hxx" -class Error; - class OggDecoder : public OggVisitor { ogg_int64_t end_granulepos; @@ -58,7 +56,7 @@ protected: return end_granulepos > 0; } - bool SeekGranulePos(ogg_int64_t where_granulepos, Error &error); + void SeekGranulePos(ogg_int64_t where_granulepos); }; #endif diff --git a/src/decoder/plugins/OpusDecoderPlugin.cxx b/src/decoder/plugins/OpusDecoderPlugin.cxx index 25e08f8e1..17dd58d48 100644 --- a/src/decoder/plugins/OpusDecoderPlugin.cxx +++ b/src/decoder/plugins/OpusDecoderPlugin.cxx @@ -261,7 +261,12 @@ MPDOpusDecoder::Seek(uint64_t where_frame) const ogg_int64_t where_granulepos(where_frame); - return SeekGranulePos(where_granulepos, IgnoreError()); + try { + SeekGranulePos(where_granulepos); + return true; + } catch (const std::runtime_error &) { + return false; + } } static void @@ -273,7 +278,10 @@ mpd_opus_stream_decode(Decoder &decoder, /* rewind the stream, because ogg_codec_detect() has moved it */ - input_stream.LockRewind(IgnoreError()); + try { + input_stream.LockRewind(); + } catch (const std::runtime_error &) { + } DecoderReader reader(decoder, input_stream); diff --git a/src/decoder/plugins/PcmDecoderPlugin.cxx b/src/decoder/plugins/PcmDecoderPlugin.cxx index 3a3adef37..3a826b086 100644 --- a/src/decoder/plugins/PcmDecoderPlugin.cxx +++ b/src/decoder/plugins/PcmDecoderPlugin.cxx @@ -29,6 +29,8 @@ #include "util/MimeType.hxx" #include "Log.hxx" +#include + #include static void @@ -145,11 +147,11 @@ pcm_stream_decode(Decoder &decoder, InputStream &is) uint64_t frame = decoder_seek_where_frame(decoder); offset_type offset = frame * frame_size; - Error error; - if (is.LockSeek(offset, error)) { + try { + is.LockSeek(offset); decoder_command_finished(decoder); - } else { - LogError(error); + } catch (const std::runtime_error &e) { + LogError(e); decoder_seek_error(decoder); } diff --git a/src/decoder/plugins/SndfileDecoderPlugin.cxx b/src/decoder/plugins/SndfileDecoderPlugin.cxx index d82a93692..ec45270fb 100644 --- a/src/decoder/plugins/SndfileDecoderPlugin.cxx +++ b/src/decoder/plugins/SndfileDecoderPlugin.cxx @@ -27,6 +27,8 @@ #include "util/Domain.hxx" #include "Log.hxx" +#include + #include static constexpr Domain sndfile_domain("sndfile"); @@ -89,13 +91,13 @@ sndfile_vio_seek(sf_count_t _offset, int whence, void *user_data) return -1; } - Error error; - if (!is.LockSeek(offset, error)) { - LogError(error, "Seek failed"); + try { + is.LockSeek(offset); + return is.GetOffset(); + } catch (const std::runtime_error &e) { + LogError(e, "Seek failed"); return -1; } - - return is.GetOffset(); } static sf_count_t diff --git a/src/decoder/plugins/VorbisDecoderPlugin.cxx b/src/decoder/plugins/VorbisDecoderPlugin.cxx index 477041230..178303df8 100644 --- a/src/decoder/plugins/VorbisDecoderPlugin.cxx +++ b/src/decoder/plugins/VorbisDecoderPlugin.cxx @@ -126,11 +126,13 @@ VorbisDecoder::Seek(uint64_t where_frame) const ogg_int64_t where_granulepos(where_frame); - if (!SeekGranulePos(where_granulepos, IgnoreError())) + try { + SeekGranulePos(where_granulepos); + vorbis_synthesis_restart(&dsp); + return true; + } catch (const std::runtime_error &) { return false; - - vorbis_synthesis_restart(&dsp); - return true; + } } void @@ -311,7 +313,10 @@ vorbis_stream_decode(Decoder &decoder, /* rewind the stream, because ogg_codec_detect() has moved it */ - input_stream.LockRewind(IgnoreError()); + try { + input_stream.LockRewind(); + } catch (const std::runtime_error &) { + } DecoderReader reader(decoder, input_stream); VorbisDecoder d(reader); diff --git a/src/decoder/plugins/WavpackDecoderPlugin.cxx b/src/decoder/plugins/WavpackDecoderPlugin.cxx index 8ab06a761..7dd3ed7b0 100644 --- a/src/decoder/plugins/WavpackDecoderPlugin.cxx +++ b/src/decoder/plugins/WavpackDecoderPlugin.cxx @@ -400,7 +400,12 @@ wavpack_input_set_pos_abs(void *id, uint32_t pos) { WavpackInput &wpi = *wpin(id); - return wpi.is.LockSeek(pos, IgnoreError()) ? 0 : -1; + try { + wpi.is.LockSeek(pos); + return 0; + } catch (const std::runtime_error &) { + return -1; + } } static int @@ -429,7 +434,12 @@ wavpack_input_set_pos_rel(void *id, int32_t delta, int mode) return -1; } - return is.LockSeek(offset, IgnoreError()) ? 0 : -1; + try { + wpi.is.LockSeek(offset); + return 0; + } catch (const std::runtime_error &) { + return -1; + } } static int diff --git a/src/input/AsyncInputStream.cxx b/src/input/AsyncInputStream.cxx index a3f4eb36f..19b14b180 100644 --- a/src/input/AsyncInputStream.cxx +++ b/src/input/AsyncInputStream.cxx @@ -78,16 +78,14 @@ AsyncInputStream::Resume() } } -bool -AsyncInputStream::Check(Error &) +void +AsyncInputStream::Check() { if (postponed_exception) { auto e = std::move(postponed_exception); postponed_exception = std::exception_ptr(); std::rethrow_exception(e); } - - return true; } bool @@ -97,15 +95,15 @@ AsyncInputStream::IsEOF() (!open && buffer.IsEmpty()); } -bool -AsyncInputStream::Seek(offset_type new_offset, Error &error) +void +AsyncInputStream::Seek(offset_type new_offset) { assert(IsReady()); assert(seek_state == SeekState::NONE); if (new_offset == offset) /* no-op */ - return true; + return; if (!IsSeekable()) throw std::runtime_error("Not seekable"); @@ -127,7 +125,7 @@ AsyncInputStream::Seek(offset_type new_offset, Error &error) } if (new_offset == offset) - return true; + return; /* no: ask the implementation to seek */ @@ -139,10 +137,7 @@ AsyncInputStream::Seek(offset_type new_offset, Error &error) while (seek_state != SeekState::NONE) cond.wait(mutex); - if (!Check(error)) - return false; - - return true; + Check(); } void @@ -177,15 +172,14 @@ AsyncInputStream::IsAvailable() } size_t -AsyncInputStream::Read(void *ptr, size_t read_size, Error &error) +AsyncInputStream::Read(void *ptr, size_t read_size) { assert(!io_thread_inside()); /* wait for data */ CircularBuffer::Range r; while (true) { - if (!Check(error)) - return 0; + Check(); r = buffer.Read(); if (!r.IsEmpty() || IsEOF()) diff --git a/src/input/AsyncInputStream.hxx b/src/input/AsyncInputStream.hxx index 6e7b427e0..f2c77ec82 100644 --- a/src/input/AsyncInputStream.hxx +++ b/src/input/AsyncInputStream.hxx @@ -81,12 +81,12 @@ public: virtual ~AsyncInputStream(); /* virtual methods from InputStream */ - bool Check(Error &error) final; + void Check() final; bool IsEOF() final; - bool Seek(offset_type new_offset, Error &error) final; + void Seek(offset_type new_offset) final; Tag *ReadTag() final; bool IsAvailable() final; - size_t Read(void *ptr, size_t read_size, Error &error) final; + size_t Read(void *ptr, size_t read_size) final; protected: /** diff --git a/src/input/IcyInputStream.cxx b/src/input/IcyInputStream.cxx index 8a7d7ca4c..0febd5ea5 100644 --- a/src/input/IcyInputStream.cxx +++ b/src/input/IcyInputStream.cxx @@ -79,13 +79,13 @@ IcyInputStream::ReadTag() } size_t -IcyInputStream::Read(void *ptr, size_t read_size, Error &error) +IcyInputStream::Read(void *ptr, size_t read_size) { if (!IsEnabled()) - return ProxyInputStream::Read(ptr, read_size, error); + return ProxyInputStream::Read(ptr, read_size); while (true) { - size_t nbytes = ProxyInputStream::Read(ptr, read_size, error); + size_t nbytes = ProxyInputStream::Read(ptr, read_size); if (nbytes == 0) return 0; diff --git a/src/input/IcyInputStream.hxx b/src/input/IcyInputStream.hxx index 88bee0b76..88471d6d3 100644 --- a/src/input/IcyInputStream.hxx +++ b/src/input/IcyInputStream.hxx @@ -62,7 +62,7 @@ public: /* virtual methods from InputStream */ void Update() override; Tag *ReadTag() override; - size_t Read(void *ptr, size_t size, Error &error) override; + size_t Read(void *ptr, size_t size) override; }; #endif diff --git a/src/input/InputStream.cxx b/src/input/InputStream.cxx index 9eaf206a0..f9b25b60c 100644 --- a/src/input/InputStream.cxx +++ b/src/input/InputStream.cxx @@ -22,16 +22,17 @@ #include "thread/Cond.hxx" #include "util/StringCompare.hxx" +#include + #include InputStream::~InputStream() { } -bool -InputStream::Check(gcc_unused Error &error) +void +InputStream::Check() { - return true; } void @@ -86,25 +87,24 @@ InputStream::CheapSeeking() const return IsSeekable() && !ExpensiveSeeking(uri.c_str()); } -bool -InputStream::Seek(gcc_unused offset_type new_offset, - gcc_unused Error &error) +void +InputStream::Seek(gcc_unused offset_type new_offset) { - return false; + throw std::runtime_error("Seeking is not implemented"); } -bool -InputStream::LockSeek(offset_type _offset, Error &error) +void +InputStream::LockSeek(offset_type _offset) { const ScopeLock protect(mutex); - return Seek(_offset, error); + Seek(_offset); } -bool -InputStream::LockSkip(offset_type _offset, Error &error) +void +InputStream::LockSkip(offset_type _offset) { const ScopeLock protect(mutex); - return Skip(_offset, error); + Skip(_offset); } Tag * @@ -127,7 +127,7 @@ InputStream::IsAvailable() } size_t -InputStream::LockRead(void *ptr, size_t _size, Error &error) +InputStream::LockRead(void *ptr, size_t _size) { #if !CLANG_CHECK_VERSION(3,6) /* disabled on clang due to -Wtautological-pointer-compare */ @@ -136,28 +136,27 @@ InputStream::LockRead(void *ptr, size_t _size, Error &error) assert(_size > 0); const ScopeLock protect(mutex); - return Read(ptr, _size, error); + return Read(ptr, _size); } -bool -InputStream::ReadFull(void *_ptr, size_t _size, Error &error) +void +InputStream::ReadFull(void *_ptr, size_t _size) { uint8_t *ptr = (uint8_t *)_ptr; size_t nbytes_total = 0; while (_size > 0) { - size_t nbytes = Read(ptr + nbytes_total, _size, error); + size_t nbytes = Read(ptr + nbytes_total, _size); if (nbytes == 0) - return false; + throw std::runtime_error("Unexpected end of file"); nbytes_total += nbytes; _size -= nbytes; } - return true; } -bool -InputStream::LockReadFull(void *ptr, size_t _size, Error &error) +void +InputStream::LockReadFull(void *ptr, size_t _size) { #if !CLANG_CHECK_VERSION(3,6) /* disabled on clang due to -Wtautological-pointer-compare */ @@ -166,7 +165,7 @@ InputStream::LockReadFull(void *ptr, size_t _size, Error &error) assert(_size > 0); const ScopeLock protect(mutex); - return ReadFull(ptr, _size, error); + ReadFull(ptr, _size); } bool diff --git a/src/input/InputStream.hxx b/src/input/InputStream.hxx index d93784991..f0330cb82 100644 --- a/src/input/InputStream.hxx +++ b/src/input/InputStream.hxx @@ -154,10 +154,9 @@ public: /** * Check for errors that may have occurred in the I/O thread. - * - * @return false on error + * Throws std::runtime_error on error. */ - virtual bool Check(Error &error); + virtual void Check(); /** * Update the public attributes. Call before accessing attributes @@ -271,36 +270,38 @@ public: * * The caller must lock the mutex. * + * Throws std::runtime_error on error. + * * @param offset the relative offset */ - virtual bool Seek(offset_type offset, Error &error); + virtual void Seek(offset_type offset); /** * Wrapper for Seek() which locks and unlocks the mutex; the * caller must not be holding it already. */ - bool LockSeek(offset_type offset, Error &error); + void LockSeek(offset_type offset); /** * Rewind to the beginning of the stream. This is a wrapper * for Seek(0, error). */ - bool Rewind(Error &error) { - return Seek(0, error); + void Rewind() { + Seek(0); } - bool LockRewind(Error &error) { - return LockSeek(0, error); + void LockRewind() { + LockSeek(0); } /** * Skip input bytes. */ - bool Skip(offset_type _offset, Error &error) { - return Seek(GetOffset() + _offset, error); + void Skip(offset_type _offset) { + Seek(GetOffset() + _offset); } - bool LockSkip(offset_type _offset, Error &error); + void LockSkip(offset_type _offset); /** * Returns true if the stream has reached end-of-file. @@ -351,38 +352,46 @@ public: * * The caller must lock the mutex. * + * Throws std::runtime_error on error. + * * @param ptr the buffer to read into * @param size the maximum number of bytes to read * @return the number of bytes read */ gcc_nonnull_all - virtual size_t Read(void *ptr, size_t size, Error &error) = 0; + virtual size_t Read(void *ptr, size_t size) = 0; /** * Wrapper for Read() which locks and unlocks the mutex; * the caller must not be holding it already. + * + * Throws std::runtime_error on error. */ gcc_nonnull_all - size_t LockRead(void *ptr, size_t size, Error &error); + size_t LockRead(void *ptr, size_t size); /** * Reads the whole data from the stream into the caller-supplied buffer. * * The caller must lock the mutex. * + * Throws std::runtime_error on error. + * * @param ptr the buffer to read into * @param size the number of bytes to read * @return true if the whole data was read, false otherwise. */ gcc_nonnull_all - bool ReadFull(void *ptr, size_t size, Error &error); + void ReadFull(void *ptr, size_t size); /** * Wrapper for ReadFull() which locks and unlocks the mutex; * the caller must not be holding it already. + * + * Throws std::runtime_error on error. */ gcc_nonnull_all - bool LockReadFull(void *ptr, size_t size, Error &error); + void LockReadFull(void *ptr, size_t size); }; #endif diff --git a/src/input/Open.cxx b/src/input/Open.cxx index 53afb5a7b..4f23f14ce 100644 --- a/src/input/Open.cxx +++ b/src/input/Open.cxx @@ -63,20 +63,11 @@ InputStream::OpenReady(const char *uri, { auto is = Open(uri, mutex, cond); - bool success; - { const ScopeLock protect(mutex); is->WaitReady(); - - Error error; - success = is->Check(error); - if (!success) - throw std::runtime_error(error.GetMessage()); + is->Check(); } - if (!success) - is.reset(); - return is; } diff --git a/src/input/ProxyInputStream.cxx b/src/input/ProxyInputStream.cxx index d0d0e99bd..d79b1681c 100644 --- a/src/input/ProxyInputStream.cxx +++ b/src/input/ProxyInputStream.cxx @@ -49,10 +49,10 @@ ProxyInputStream::CopyAttributes() } } -bool -ProxyInputStream::Check(Error &error) +void +ProxyInputStream::Check() { - return input.Check(error); + input.Check(); } void @@ -62,12 +62,11 @@ ProxyInputStream::Update() CopyAttributes(); } -bool -ProxyInputStream::Seek(offset_type new_offset, Error &error) +void +ProxyInputStream::Seek(offset_type new_offset) { - bool success = input.Seek(new_offset, error); + input.Seek(new_offset); CopyAttributes(); - return success; } bool @@ -89,9 +88,9 @@ ProxyInputStream::IsAvailable() } size_t -ProxyInputStream::Read(void *ptr, size_t read_size, Error &error) +ProxyInputStream::Read(void *ptr, size_t read_size) { - size_t nbytes = input.Read(ptr, read_size, error); + size_t nbytes = input.Read(ptr, read_size); CopyAttributes(); return nbytes; } diff --git a/src/input/ProxyInputStream.hxx b/src/input/ProxyInputStream.hxx index 80b92e7d9..900484c6f 100644 --- a/src/input/ProxyInputStream.hxx +++ b/src/input/ProxyInputStream.hxx @@ -43,13 +43,13 @@ public: ProxyInputStream &operator=(const ProxyInputStream &) = delete; /* virtual methods from InputStream */ - bool Check(Error &error) override; + void Check() override; void Update() override; - bool Seek(offset_type new_offset, Error &error) override; + void Seek(offset_type new_offset) override; bool IsEOF() override; Tag *ReadTag() override; bool IsAvailable() override; - size_t Read(void *ptr, size_t read_size, Error &error) override; + size_t Read(void *ptr, size_t read_size) override; protected: /** diff --git a/src/input/Reader.cxx b/src/input/Reader.cxx index 0401e0fe9..c97b0f15c 100644 --- a/src/input/Reader.cxx +++ b/src/input/Reader.cxx @@ -20,19 +20,12 @@ #include "config.h" #include "Reader.hxx" #include "InputStream.hxx" -#include "util/Error.hxx" -#include "Log.hxx" size_t InputStreamReader::Read(void *data, size_t size) { - Error error; - size_t nbytes = is.LockRead(data, size, error); - assert(nbytes == 0 || !error.IsDefined()); - assert(nbytes > 0 || error.IsDefined() || is.IsEOF()); - - if (gcc_unlikely(nbytes == 0 && error.IsDefined())) - LogError(error); + size_t nbytes = is.LockRead(data, size); + assert(nbytes > 0 || is.IsEOF()); return nbytes; } diff --git a/src/input/TextInputStream.cxx b/src/input/TextInputStream.cxx index 74457129a..afa50fe94 100644 --- a/src/input/TextInputStream.cxx +++ b/src/input/TextInputStream.cxx @@ -20,10 +20,11 @@ #include "config.h" #include "TextInputStream.hxx" #include "InputStream.hxx" -#include "util/Error.hxx" #include "util/TextFile.hxx" #include "Log.hxx" +#include + #include TextInputStream::TextInputStream(InputStreamPtr &&_is) @@ -58,12 +59,12 @@ TextInputStream::ReadLine() character */ --dest.size; - Error error; - size_t nbytes = is->LockRead(dest.data, dest.size, error); - if (nbytes > 0) - buffer.Append(nbytes); - else if (error.IsDefined()) { - LogError(error); + size_t nbytes; + + try { + nbytes = is->LockRead(dest.data, dest.size); + } catch (const std::runtime_error &e) { + LogError(e); return nullptr; } diff --git a/src/input/ThreadInputStream.cxx b/src/input/ThreadInputStream.cxx index a637102e7..56054e15b 100644 --- a/src/input/ThreadInputStream.cxx +++ b/src/input/ThreadInputStream.cxx @@ -114,15 +114,13 @@ ThreadInputStream::ThreadFunc(void *ctx) tis.ThreadFunc(); } -bool -ThreadInputStream::Check(Error &) +void +ThreadInputStream::Check() { assert(!thread.IsInside()); if (postponed_exception) std::rethrow_exception(postponed_exception); - - return true; } bool @@ -134,7 +132,7 @@ ThreadInputStream::IsAvailable() } inline size_t -ThreadInputStream::Read(void *ptr, size_t read_size, Error &) +ThreadInputStream::Read(void *ptr, size_t read_size) { assert(!thread.IsInside()); diff --git a/src/input/ThreadInputStream.hxx b/src/input/ThreadInputStream.hxx index 5001c070d..31ed497b6 100644 --- a/src/input/ThreadInputStream.hxx +++ b/src/input/ThreadInputStream.hxx @@ -83,10 +83,10 @@ public: void Start(); /* virtual methods from InputStream */ - bool Check(Error &error) override final; + void Check() override final; bool IsEOF() override final; bool IsAvailable() override final; - size_t Read(void *ptr, size_t size, Error &error) override final; + size_t Read(void *ptr, size_t size) override final; protected: void SetMimeType(const char *_mime) { diff --git a/src/input/plugins/CdioParanoiaInputPlugin.cxx b/src/input/plugins/CdioParanoiaInputPlugin.cxx index f06d01a10..832363f82 100644 --- a/src/input/plugins/CdioParanoiaInputPlugin.cxx +++ b/src/input/plugins/CdioParanoiaInputPlugin.cxx @@ -98,8 +98,8 @@ class CdioParanoiaInputStream final : public InputStream { /* virtual methods from InputStream */ bool IsEOF() override; - size_t Read(void *ptr, size_t size, Error &error) override; - bool Seek(offset_type offset, Error &error) override; + size_t Read(void *ptr, size_t size) override; + void Seek(offset_type offset) override; }; static constexpr Domain cdio_domain("cdio"); @@ -256,30 +256,26 @@ input_cdio_open(const char *uri, lsn_from, lsn_to); } -bool -CdioParanoiaInputStream::Seek(offset_type new_offset, Error &error) +void +CdioParanoiaInputStream::Seek(offset_type new_offset) { - if (new_offset > size) { - error.Format(cdio_domain, "Invalid offset to seek %ld (%ld)", - (long int)new_offset, (long int)size); - return false; - } + if (new_offset > size) + throw FormatRuntimeError("Invalid offset to seek %ld (%ld)", + (long int)new_offset, (long int)size); /* simple case */ if (new_offset == offset) - return true; + return; /* calculate current LSN */ lsn_relofs = new_offset / CDIO_CD_FRAMESIZE_RAW; offset = new_offset; cdio_paranoia_seek(para, lsn_from + lsn_relofs, SEEK_SET); - - return true; } size_t -CdioParanoiaInputStream::Read(void *ptr, size_t length, Error &error) +CdioParanoiaInputStream::Read(void *ptr, size_t length) { size_t nbytes = 0; int diff; @@ -309,11 +305,9 @@ CdioParanoiaInputStream::Read(void *ptr, size_t length, Error &error) if (s_mess) { free(s_mess); } - if (!rbuf) { - error.Set(cdio_domain, - "paranoia read error. Stopping."); - return 0; - } + if (!rbuf) + throw std::runtime_error("paranoia read error"); + //store current buffer memcpy(buffer, rbuf, CDIO_CD_FRAMESIZE_RAW); buffer_lsn = lsn_relofs; diff --git a/src/input/plugins/FfmpegInputPlugin.cxx b/src/input/plugins/FfmpegInputPlugin.cxx index ab189c490..d169eb640 100644 --- a/src/input/plugins/FfmpegInputPlugin.cxx +++ b/src/input/plugins/FfmpegInputPlugin.cxx @@ -61,8 +61,8 @@ struct FfmpegInputStream final : public InputStream { /* virtual methods from InputStream */ bool IsEOF() override; - size_t Read(void *ptr, size_t size, Error &error) override; - bool Seek(offset_type offset, Error &error) override; + size_t Read(void *ptr, size_t size) override; + void Seek(offset_type offset) override; }; static inline bool @@ -103,15 +103,15 @@ input_ffmpeg_open(const char *uri, } size_t -FfmpegInputStream::Read(void *ptr, size_t read_size, Error &error) +FfmpegInputStream::Read(void *ptr, size_t read_size) { auto result = avio_read(h, (unsigned char *)ptr, read_size); if (result <= 0) { if (result < 0) - SetFfmpegError(error, result, "avio_read() failed"); + throw MakeFfmpegError(result, "avio_read() failed"); eof = true; - return false; + return 0; } offset += result; @@ -124,19 +124,16 @@ FfmpegInputStream::IsEOF() return eof; } -bool -FfmpegInputStream::Seek(offset_type new_offset, Error &error) +void +FfmpegInputStream::Seek(offset_type new_offset) { auto result = avio_seek(h, new_offset, SEEK_SET); - if (result < 0) { - SetFfmpegError(error, result, "avio_seek() failed"); - return false; - } + if (result < 0) + throw MakeFfmpegError(result, "avio_seek() failed"); offset = result; eof = false; - return true; } const InputPlugin input_plugin_ffmpeg = { diff --git a/src/input/plugins/FileInputPlugin.cxx b/src/input/plugins/FileInputPlugin.cxx index 6fb816423..6879946dc 100644 --- a/src/input/plugins/FileInputPlugin.cxx +++ b/src/input/plugins/FileInputPlugin.cxx @@ -49,8 +49,8 @@ public: return GetOffset() >= GetSize(); } - size_t Read(void *ptr, size_t size, Error &error) override; - bool Seek(offset_type offset, Error &error) override; + size_t Read(void *ptr, size_t size) override; + void Seek(offset_type offset) override; }; InputStreamPtr @@ -84,26 +84,19 @@ input_file_open(gcc_unused const char *filename, return nullptr; } -bool -FileInputStream::Seek(offset_type new_offset, Error &error) -try { +void +FileInputStream::Seek(offset_type new_offset) +{ reader.Seek((off_t)new_offset); offset = new_offset; - return true; -} catch (const std::exception &e) { - error.Set(std::current_exception()); - return false; } size_t -FileInputStream::Read(void *ptr, size_t read_size, Error &error) -try { +FileInputStream::Read(void *ptr, size_t read_size) +{ size_t nbytes = reader.Read(ptr, read_size); offset += nbytes; return nbytes; -} catch (const std::exception &e) { - error.Set(std::current_exception()); - return 0; } const InputPlugin input_plugin_file = { diff --git a/src/input/plugins/RewindInputPlugin.cxx b/src/input/plugins/RewindInputPlugin.cxx index 01f269056..08f50917c 100644 --- a/src/input/plugins/RewindInputPlugin.cxx +++ b/src/input/plugins/RewindInputPlugin.cxx @@ -63,8 +63,8 @@ public: return !ReadingFromBuffer() && ProxyInputStream::IsEOF(); } - size_t Read(void *ptr, size_t size, Error &error) override; - bool Seek(offset_type offset, Error &error) override; + size_t Read(void *ptr, size_t size) override; + void Seek(offset_type offset) override; private: /** @@ -77,7 +77,7 @@ private: }; size_t -RewindInputStream::Read(void *ptr, size_t read_size, Error &error) +RewindInputStream::Read(void *ptr, size_t read_size) { if (ReadingFromBuffer()) { /* buffered read */ @@ -96,7 +96,7 @@ RewindInputStream::Read(void *ptr, size_t read_size, Error &error) } else { /* pass method call to underlying stream */ - size_t nbytes = input.Read(ptr, read_size, error); + size_t nbytes = input.Read(ptr, read_size); if (input.GetOffset() > (offset_type)sizeof(buffer)) /* disable buffering */ @@ -116,9 +116,8 @@ RewindInputStream::Read(void *ptr, size_t read_size, Error &error) } } -bool -RewindInputStream::Seek(offset_type new_offset, - Error &error) +void +RewindInputStream::Seek(offset_type new_offset) { assert(IsReady()); @@ -131,14 +130,12 @@ RewindInputStream::Seek(offset_type new_offset, head = (size_t)new_offset; offset = new_offset; - - return true; } else { /* disable the buffer, because input has left the buffered range now */ tail = 0; - return ProxyInputStream::Seek(new_offset, error); + ProxyInputStream::Seek(new_offset); } } diff --git a/src/input/plugins/SmbclientInputPlugin.cxx b/src/input/plugins/SmbclientInputPlugin.cxx index 7eea671f0..641a15c8f 100644 --- a/src/input/plugins/SmbclientInputPlugin.cxx +++ b/src/input/plugins/SmbclientInputPlugin.cxx @@ -26,7 +26,6 @@ #include "PluginUnavailable.hxx" #include "system/Error.hxx" #include "util/StringCompare.hxx" -#include "util/Error.hxx" #include @@ -60,8 +59,8 @@ public: return offset >= size; } - size_t Read(void *ptr, size_t size, Error &error) override; - bool Seek(offset_type offset, Error &error) override; + size_t Read(void *ptr, size_t size) override; + void Seek(offset_type offset) override; }; /* @@ -124,33 +123,28 @@ input_smbclient_open(const char *uri, } size_t -SmbclientInputStream::Read(void *ptr, size_t read_size, Error &error) +SmbclientInputStream::Read(void *ptr, size_t read_size) { smbclient_mutex.lock(); ssize_t nbytes = smbc_read(fd, ptr, read_size); smbclient_mutex.unlock(); - if (nbytes < 0) { - error.SetErrno("smbc_read() failed"); - nbytes = 0; - } + if (nbytes < 0) + throw MakeErrno("smbc_read() failed"); offset += nbytes; return nbytes; } -bool -SmbclientInputStream::Seek(offset_type new_offset, Error &error) +void +SmbclientInputStream::Seek(offset_type new_offset) { smbclient_mutex.lock(); off_t result = smbc_lseek(fd, new_offset, SEEK_SET); smbclient_mutex.unlock(); - if (result < 0) { - error.SetErrno("smbc_lseek() failed"); - return false; - } + if (result < 0) + throw MakeErrno("smbc_lseek() failed"); offset = result; - return true; } const InputPlugin input_plugin_smbclient = { diff --git a/src/lib/expat/ExpatParser.cxx b/src/lib/expat/ExpatParser.cxx index 13d16707d..99050f886 100644 --- a/src/lib/expat/ExpatParser.cxx +++ b/src/lib/expat/ExpatParser.cxx @@ -43,25 +43,21 @@ ExpatParser::Parse(const char *data, size_t length, bool is_final) throw ExpatError(parser); } -bool -ExpatParser::Parse(InputStream &is, Error &error) +void +ExpatParser::Parse(InputStream &is) { assert(is.IsReady()); while (true) { char buffer[4096]; - size_t nbytes = is.LockRead(buffer, sizeof(buffer), error); + size_t nbytes = is.LockRead(buffer, sizeof(buffer)); if (nbytes == 0) break; Parse(buffer, nbytes, false); } - if (error.IsDefined()) - return false; - Parse("", 0, true); - return true; } const char * diff --git a/src/lib/expat/ExpatParser.hxx b/src/lib/expat/ExpatParser.hxx index 15983801c..c13017fb3 100644 --- a/src/lib/expat/ExpatParser.hxx +++ b/src/lib/expat/ExpatParser.hxx @@ -66,7 +66,7 @@ public: void Parse(const char *data, size_t length, bool is_final); - bool Parse(InputStream &is, Error &error); + void Parse(InputStream &is); gcc_pure static const char *GetAttribute(const XML_Char **atts, @@ -97,8 +97,8 @@ public: parser.Parse(data, length, is_final); } - bool Parse(InputStream &is, Error &error) { - return parser.Parse(is, error); + void Parse(InputStream &is) { + parser.Parse(is); } gcc_pure diff --git a/src/lib/ffmpeg/Error.cxx b/src/lib/ffmpeg/Error.cxx index 92608300f..2b9cc5a6c 100644 --- a/src/lib/ffmpeg/Error.cxx +++ b/src/lib/ffmpeg/Error.cxx @@ -21,6 +21,7 @@ #include "Error.hxx" #include "Domain.hxx" #include "util/Error.hxx" +#include "util/RuntimeError.hxx" extern "C" { #include @@ -34,6 +35,14 @@ MakeFfmpegError(int errnum) return std::runtime_error(msg); } +std::runtime_error +MakeFfmpegError(int errnum, const char *prefix) +{ + char msg[256]; + av_strerror(errnum, msg, sizeof(msg)); + return FormatRuntimeError("%s: %s", prefix, msg); +} + void SetFfmpegError(Error &error, int errnum) { diff --git a/src/lib/ffmpeg/Error.hxx b/src/lib/ffmpeg/Error.hxx index 3a6eb876d..028dffbaa 100644 --- a/src/lib/ffmpeg/Error.hxx +++ b/src/lib/ffmpeg/Error.hxx @@ -27,6 +27,9 @@ class Error; std::runtime_error MakeFfmpegError(int errnum); +std::runtime_error +MakeFfmpegError(int errnum, const char *prefix); + void SetFfmpegError(Error &error, int errnum); diff --git a/src/lib/xiph/OggFind.cxx b/src/lib/xiph/OggFind.cxx index 994cfa403..e9b9b8304 100644 --- a/src/lib/xiph/OggFind.cxx +++ b/src/lib/xiph/OggFind.cxx @@ -21,7 +21,8 @@ #include "OggFind.hxx" #include "lib/xiph/OggSyncState.hxx" #include "input/InputStream.hxx" -#include "util/Error.hxx" + +#include bool OggFindEOS(OggSyncState &oy, ogg_stream_state &os, ogg_packet &packet) @@ -48,8 +49,13 @@ OggSeekPageAtOffset(OggSyncState &oy, ogg_stream_state &os, InputStream &is, data */ ogg_stream_reset(&os); - return is.LockSeek(offset, IgnoreError()) && - oy.ExpectPageSeekIn(os); + try { + is.LockSeek(offset); + } catch (const std::runtime_error &) { + return false; + } + + return oy.ExpectPageSeekIn(os); } bool diff --git a/src/playlist/PlaylistRegistry.cxx b/src/playlist/PlaylistRegistry.cxx index 6e8330f10..d60884519 100644 --- a/src/playlist/PlaylistRegistry.cxx +++ b/src/playlist/PlaylistRegistry.cxx @@ -34,11 +34,12 @@ #include "util/MimeType.hxx" #include "util/UriUtil.hxx" #include "util/StringUtil.hxx" -#include "util/Error.hxx" #include "util/Macros.hxx" #include "config/ConfigGlobal.hxx" #include "config/Block.hxx" +#include + #include #include @@ -193,7 +194,10 @@ playlist_list_open_stream_mime2(InputStreamPtr &&is, const char *mime) StringArrayContainsCase(plugin->mime_types, mime)) { /* rewind the stream, so each plugin gets a fresh start */ - is->Rewind(IgnoreError()); + try { + is->Rewind(); + } catch (const std::runtime_error &) { + } auto playlist = playlist_plugin_open_stream(plugin, std::move(is)); @@ -234,7 +238,10 @@ playlist_list_open_stream_suffix(InputStreamPtr &&is, const char *suffix) StringArrayContainsCase(plugin->suffixes, suffix)) { /* rewind the stream, so each plugin gets a fresh start */ - is->Rewind(IgnoreError()); + try { + is->Rewind(); + } catch (const std::runtime_error &) { + } auto playlist = playlist_plugin_open_stream(plugin, std::move(is)); diff --git a/src/playlist/plugins/AsxPlaylistPlugin.cxx b/src/playlist/plugins/AsxPlaylistPlugin.cxx index 158f0d5eb..483a43ecb 100644 --- a/src/playlist/plugins/AsxPlaylistPlugin.cxx +++ b/src/playlist/plugins/AsxPlaylistPlugin.cxx @@ -23,7 +23,6 @@ #include "../MemorySongEnumerator.hxx" #include "tag/TagBuilder.hxx" #include "util/ASCII.hxx" -#include "util/Error.hxx" #include "util/StringView.hxx" #include "lib/expat/ExpatParser.hxx" #include "Log.hxx" @@ -152,12 +151,7 @@ asx_open_stream(InputStreamPtr &&is) ExpatParser expat(&parser); expat.SetElementHandler(asx_start_element, asx_end_element); expat.SetCharacterDataHandler(asx_char_data); - - Error error; - if (!expat.Parse(*is, error)) { - LogError(error); - return nullptr; - } + expat.Parse(*is); } parser.songs.reverse(); diff --git a/src/playlist/plugins/RssPlaylistPlugin.cxx b/src/playlist/plugins/RssPlaylistPlugin.cxx index d481a2882..3b774b5db 100644 --- a/src/playlist/plugins/RssPlaylistPlugin.cxx +++ b/src/playlist/plugins/RssPlaylistPlugin.cxx @@ -23,7 +23,6 @@ #include "../MemorySongEnumerator.hxx" #include "tag/TagBuilder.hxx" #include "util/ASCII.hxx" -#include "util/Error.hxx" #include "util/StringView.hxx" #include "lib/expat/ExpatParser.hxx" #include "Log.hxx" @@ -150,12 +149,7 @@ rss_open_stream(InputStreamPtr &&is) ExpatParser expat(&parser); expat.SetElementHandler(rss_start_element, rss_end_element); expat.SetCharacterDataHandler(rss_char_data); - - Error error; - if (!expat.Parse(*is, error)) { - LogError(error); - return nullptr; - } + expat.Parse(*is); } parser.songs.reverse(); diff --git a/src/playlist/plugins/SoundCloudPlaylistPlugin.cxx b/src/playlist/plugins/SoundCloudPlaylistPlugin.cxx index 67a64ce2e..fceb13ec5 100644 --- a/src/playlist/plugins/SoundCloudPlaylistPlugin.cxx +++ b/src/playlist/plugins/SoundCloudPlaylistPlugin.cxx @@ -26,7 +26,6 @@ #include "tag/TagBuilder.hxx" #include "util/StringCompare.hxx" #include "util/Alloc.hxx" -#include "util/Error.hxx" #include "util/Domain.hxx" #include "util/ScopeExit.hxx" #include "Log.hxx" @@ -231,29 +230,20 @@ static int soundcloud_parse_json(const char *url, yajl_handle hand, Mutex &mutex, Cond &cond) try { - Error error; auto input_stream = InputStream::OpenReady(url, mutex, cond); const ScopeLock protect(mutex); yajl_status stat; - int done = 0; + bool done = false; while (!done) { char buffer[4096]; unsigned char *ubuffer = (unsigned char *)buffer; const size_t nbytes = - input_stream->Read(buffer, sizeof(buffer), error); - if (nbytes == 0) { - if (error.IsDefined()) - LogError(error); - - if (input_stream->IsEOF()) { - done = true; - } else { - return -1; - } - } + input_stream->Read(buffer, sizeof(buffer)); + if (nbytes == 0) + done = true; if (done) { stat = yajl_complete_parse(hand); diff --git a/src/playlist/plugins/XspfPlaylistPlugin.cxx b/src/playlist/plugins/XspfPlaylistPlugin.cxx index ee29febc1..58dc949b1 100644 --- a/src/playlist/plugins/XspfPlaylistPlugin.cxx +++ b/src/playlist/plugins/XspfPlaylistPlugin.cxx @@ -24,7 +24,6 @@ #include "DetachedSong.hxx" #include "input/InputStream.hxx" #include "tag/TagBuilder.hxx" -#include "util/Error.hxx" #include "util/StringView.hxx" #include "lib/expat/ExpatParser.hxx" #include "Log.hxx" @@ -197,12 +196,7 @@ xspf_open_stream(InputStreamPtr &&is) ExpatParser expat(&parser); expat.SetElementHandler(xspf_start_element, xspf_end_element); expat.SetCharacterDataHandler(xspf_char_data); - - Error error; - if (!expat.Parse(*is, error)) { - LogError(error); - return nullptr; - } + expat.Parse(*is); } parser.songs.reverse(); diff --git a/src/tag/Aiff.cxx b/src/tag/Aiff.cxx index ccedafb0a..8f56ed2f3 100644 --- a/src/tag/Aiff.cxx +++ b/src/tag/Aiff.cxx @@ -21,10 +21,9 @@ #include "Aiff.hxx" #include "input/InputStream.hxx" #include "system/ByteOrder.hxx" -#include "Log.hxx" -#include "util/Error.hxx" #include +#include #include #include @@ -45,33 +44,27 @@ aiff_seek_id3(InputStream &is) { /* seek to the beginning and read the AIFF header */ - Error error; - if (!is.Rewind(error)) { - LogError(error, "Failed to seek"); - return 0; - } + is.Rewind(); aiff_header header; - if (!is.ReadFull(&header, sizeof(header), IgnoreError()) || - memcmp(header.id, "FORM", 4) != 0 || + is.ReadFull(&header, sizeof(header)); + if (memcmp(header.id, "FORM", 4) != 0 || (is.KnownSize() && FromLE32(header.size) > is.GetSize()) || (memcmp(header.format, "AIFF", 4) != 0 && memcmp(header.format, "AIFC", 4) != 0)) - /* not a AIFF file */ - return 0; + throw std::runtime_error("Not an AIFF file"); while (true) { /* read the chunk header */ aiff_chunk_header chunk; - if (!is.ReadFull(&chunk, sizeof(chunk), IgnoreError())) - return 0; + is.ReadFull(&chunk, sizeof(chunk)); size_t size = FromBE32(chunk.size); if (size > size_t(std::numeric_limits::max())) /* too dangerous, bail out: possible integer underflow when casting to off_t */ - return 0; + throw std::runtime_error("AIFF chunk is too large"); if (memcmp(chunk.id, "ID3 ", 4) == 0) /* found it! */ @@ -81,7 +74,6 @@ aiff_seek_id3(InputStream &is) /* pad byte */ ++size; - if (!is.Skip(size, IgnoreError())) - return 0; + is.Skip(size); } } diff --git a/src/tag/Aiff.hxx b/src/tag/Aiff.hxx index cb5be8b20..07dc17451 100644 --- a/src/tag/Aiff.hxx +++ b/src/tag/Aiff.hxx @@ -32,9 +32,10 @@ class InputStream; /** * Seeks the AIFF file to the ID3 chunk. * + * Throws std::runtime_error on error. + * * @param is a locked #InputStream - * @return the size of the ID3 chunk on success, or 0 if this is not a - * AIFF file or no ID3 chunk was found + * @return the size of the ID3 chunk */ size_t aiff_seek_id3(InputStream &is); diff --git a/src/tag/ApeLoader.cxx b/src/tag/ApeLoader.cxx index 8f064b25c..1ef8c4067 100644 --- a/src/tag/ApeLoader.cxx +++ b/src/tag/ApeLoader.cxx @@ -50,9 +50,10 @@ try { /* determine if file has an apeV2 tag */ ApeFooter footer; - if (!is.Seek(is.GetSize() - sizeof(footer), IgnoreError()) || - !is.ReadFull(&footer, sizeof(footer), IgnoreError()) || - memcmp(footer.id, "APETAGEX", sizeof(footer.id)) != 0 || + is.Seek(is.GetSize() - sizeof(footer)); + is.ReadFull(&footer, sizeof(footer)); + + if (memcmp(footer.id, "APETAGEX", sizeof(footer.id)) != 0 || FromLE32(footer.version) != 2000) return false; @@ -60,17 +61,17 @@ try { size_t remaining = FromLE32(footer.length); if (remaining <= sizeof(footer) + 10 || /* refuse to load more than one megabyte of tag data */ - remaining > 1024 * 1024 || - !is.Seek(is.GetSize() - remaining, IgnoreError())) + remaining > 1024 * 1024) return false; + is.Seek(is.GetSize() - remaining); + /* read tag into buffer */ remaining -= sizeof(footer); assert(remaining > 10); std::unique_ptr buffer(new char[remaining]); - if (!is.ReadFull(buffer.get(), remaining, IgnoreError())) - return false; + is.ReadFull(buffer.get(), remaining); /* read tags */ unsigned n = FromLE32(footer.count); diff --git a/src/tag/Generic.cxx b/src/tag/Generic.cxx index ad2f1e6e1..303a8ac55 100644 --- a/src/tag/Generic.cxx +++ b/src/tag/Generic.cxx @@ -38,8 +38,11 @@ ScanGenericTags(InputStream &is, const TagHandler &handler, void *ctx) return true; #ifdef ENABLE_ID3TAG - if (!is.LockRewind(IgnoreError())) + try { + is.LockRewind(); + } catch (const std::runtime_error &) { return false; + } return tag_id3_scan(is, handler, ctx); #else diff --git a/src/tag/Id3Load.cxx b/src/tag/Id3Load.cxx index bdc777712..f443e4207 100644 --- a/src/tag/Id3Load.cxx +++ b/src/tag/Id3Load.cxx @@ -19,8 +19,6 @@ #include "config.h" #include "Id3Load.hxx" -#include "util/Error.hxx" -#include "util/Domain.hxx" #include "Log.hxx" #include "Riff.hxx" #include "Aiff.hxx" @@ -31,8 +29,6 @@ #include #include -static constexpr Domain id3_domain("id3"); - static constexpr size_t ID3V1_SIZE = 128; gcc_pure @@ -46,11 +42,8 @@ static long get_id3v2_footer_size(InputStream &is, offset_type offset) try { id3_byte_t buf[ID3_TAG_QUERYSIZE]; - if (!is.Seek(offset, IgnoreError())) - return 0; - - if (!is.ReadFull(buf, sizeof(buf), IgnoreError())) - return 0; + is.Seek(offset); + is.ReadFull(buf, sizeof(buf)); return id3_tag_query(buf, sizeof(buf)); } catch (const std::runtime_error &) { @@ -61,8 +54,7 @@ static UniqueId3Tag ReadId3Tag(InputStream &is) try { id3_byte_t query_buffer[ID3_TAG_QUERYSIZE]; - if (!is.ReadFull(query_buffer, sizeof(query_buffer), IgnoreError())) - return nullptr; + is.ReadFull(query_buffer, sizeof(query_buffer)); /* Look for a tag header */ long tag_size = id3_tag_query(query_buffer, sizeof(query_buffer)); @@ -82,9 +74,7 @@ try { /* now read the remaining bytes */ const size_t remaining = tag_size - sizeof(query_buffer); - const size_t nbytes = is.Read(end, remaining, IgnoreError()); - if (nbytes != remaining) - return nullptr; + is.ReadFull(end, remaining); return UniqueId3Tag(id3_tag_parse(tag_buffer.get(), tag_size)); } catch (const std::runtime_error &) { @@ -94,8 +84,7 @@ try { static UniqueId3Tag ReadId3Tag(InputStream &is, offset_type offset) try { - if (!is.Seek(offset, IgnoreError())) - return nullptr; + is.Seek(offset); return ReadId3Tag(is); } catch (const std::runtime_error &) { @@ -106,9 +95,7 @@ static UniqueId3Tag ReadId3v1Tag(InputStream &is) try { id3_byte_t buffer[ID3V1_SIZE]; - - if (is.Read(buffer, ID3V1_SIZE, IgnoreError()) != ID3V1_SIZE) - return nullptr; + is.ReadFull(buffer, ID3V1_SIZE); return UniqueId3Tag(id3_tag_parse(buffer, ID3V1_SIZE)); } catch (const std::runtime_error &) { @@ -118,9 +105,7 @@ try { static UniqueId3Tag ReadId3v1Tag(InputStream &is, offset_type offset) try { - if (!is.Seek(offset, IgnoreError())) - return nullptr; - + is.Seek(offset); return ReadId3v1Tag(is); } catch (const std::runtime_error &) { return nullptr; @@ -203,21 +188,19 @@ try { static UniqueId3Tag tag_id3_riff_aiff_load(InputStream &is) try { - size_t size = riff_seek_id3(is); - if (size == 0) + size_t size; + try { + size = riff_seek_id3(is); + } catch (const std::runtime_error &) { size = aiff_seek_id3(is); - if (size == 0) - return nullptr; + } if (size > 4 * 1024 * 1024) /* too large, don't allocate so much memory */ return nullptr; std::unique_ptr buffer(new id3_byte_t[size]); - if (!is.ReadFull(buffer.get(), size, IgnoreError())) { - LogWarning(id3_domain, "Failed to read RIFF chunk"); - return nullptr; - } + is.ReadFull(buffer.get(), size); return UniqueId3Tag(id3_tag_parse(buffer.get(), size)); } catch (const std::runtime_error &) { diff --git a/src/tag/Riff.cxx b/src/tag/Riff.cxx index 38f8df5f7..9420c599f 100644 --- a/src/tag/Riff.cxx +++ b/src/tag/Riff.cxx @@ -21,10 +21,9 @@ #include "Riff.hxx" #include "input/InputStream.hxx" #include "system/ByteOrder.hxx" -#include "Log.hxx" -#include "util/Error.hxx" #include +#include #include #include @@ -45,31 +44,25 @@ riff_seek_id3(InputStream &is) { /* seek to the beginning and read the RIFF header */ - Error error; - if (!is.Rewind(error)) { - LogError(error, "Failed to seek"); - return 0; - } + is.Rewind(); riff_header header; - if (!is.ReadFull(&header, sizeof(header), IgnoreError()) || - memcmp(header.id, "RIFF", 4) != 0 || + is.ReadFull(&header, sizeof(header)); + if (memcmp(header.id, "RIFF", 4) != 0 || (is.KnownSize() && FromLE32(header.size) > is.GetSize())) - /* not a RIFF file */ - return 0; + throw std::runtime_error("Not a RIFF file"); while (true) { /* read the chunk header */ riff_chunk_header chunk; - if (!is.ReadFull(&chunk, sizeof(chunk), IgnoreError())) - return 0; + is.ReadFull(&chunk, sizeof(chunk)); size_t size = FromLE32(chunk.size); if (size > size_t(std::numeric_limits::max())) /* too dangerous, bail out: possible integer underflow when casting to off_t */ - return 0; + throw std::runtime_error("RIFF chunk is too large"); if (memcmp(chunk.id, "id3 ", 4) == 0 || memcmp(chunk.id, "ID3 ", 4) == 0) @@ -80,7 +73,6 @@ riff_seek_id3(InputStream &is) /* pad byte */ ++size; - if (!is.Skip(size, IgnoreError())) - return 0; + is.Skip(size); } } diff --git a/src/tag/Riff.hxx b/src/tag/Riff.hxx index b7da8bc14..8e2c02263 100644 --- a/src/tag/Riff.hxx +++ b/src/tag/Riff.hxx @@ -32,9 +32,10 @@ class InputStream; /** * Seeks the RIFF file to the ID3 chunk. * + * Throws std::runtime_error on error. + * * @param is a locked #InputStream - * @return the size of the ID3 chunk on success, or 0 if this is not a - * RIFF file or no ID3 chunk was found + * @return the size of the ID3 chunk */ size_t riff_seek_id3(InputStream &is); diff --git a/test/DumpDatabase.cxx b/test/DumpDatabase.cxx index e673dc698..2edf5d13a 100644 --- a/test/DumpDatabase.cxx +++ b/test/DumpDatabase.cxx @@ -36,6 +36,7 @@ #include "util/Error.hxx" #include "util/ScopeExit.hxx" +#include #include using std::cout; using std::cerr; @@ -46,7 +47,7 @@ using std::endl; #ifdef ENABLE_UPNP #include "input/InputStream.hxx" size_t -InputStream::LockRead(void *, size_t, Error &) +InputStream::LockRead(void *, size_t) { return 0; } diff --git a/test/FakeDecoderAPI.cxx b/test/FakeDecoderAPI.cxx index d2dffcf18..fbd848fac 100644 --- a/test/FakeDecoderAPI.cxx +++ b/test/FakeDecoderAPI.cxx @@ -21,9 +21,10 @@ #include "FakeDecoderAPI.hxx" #include "decoder/DecoderAPI.hxx" #include "input/InputStream.hxx" -#include "util/Error.hxx" #include "Compiler.h" +#include + #include #include @@ -84,7 +85,11 @@ decoder_read(gcc_unused Decoder *decoder, InputStream &is, void *buffer, size_t length) { - return is.LockRead(buffer, length, IgnoreError()); + try { + return is.LockRead(buffer, length); + } catch (const std::runtime_error &) { + return 0; + } } bool diff --git a/test/dump_text_file.cxx b/test/dump_text_file.cxx index 512a13102..918d0a721 100644 --- a/test/dump_text_file.cxx +++ b/test/dump_text_file.cxx @@ -23,7 +23,6 @@ #include "input/InputStream.hxx" #include "input/TextInputStream.hxx" #include "config/ConfigGlobal.hxx" -#include "util/Error.hxx" #include "thread/Cond.hxx" #include "Log.hxx" @@ -55,12 +54,7 @@ dump_input_stream(InputStreamPtr &&is) const ScopeLock protect(is->mutex); - Error error; - if (!is->Check(error)) { - LogError(error); - return EXIT_FAILURE; - } - + is->Check(); return 0; } diff --git a/test/run_input.cxx b/test/run_input.cxx index 675f0ced3..b7212b94c 100644 --- a/test/run_input.cxx +++ b/test/run_input.cxx @@ -24,7 +24,6 @@ #include "input/InputStream.hxx" #include "input/Init.hxx" #include "ScopeIOThread.hxx" -#include "util/Error.hxx" #include "thread/Cond.hxx" #include "Log.hxx" #include "fs/io/BufferedOutputStream.hxx" @@ -68,26 +67,17 @@ dump_input_stream(InputStream *is) delete tag; } - Error error; char buffer[4096]; - size_t num_read = is->Read(buffer, sizeof(buffer), error); - if (num_read == 0) { - if (error.IsDefined()) - LogError(error); - + size_t num_read = is->Read(buffer, sizeof(buffer)); + if (num_read == 0) break; - } ssize_t num_written = write(1, buffer, num_read); if (num_written <= 0) break; } - Error error; - if (!is->Check(error)) { - LogError(error); - return EXIT_FAILURE; - } + is->Check(); return 0; } @@ -109,7 +99,6 @@ try { archive_plugin_init_all(); #endif - Error error; input_stream_global_init(); /* open the stream and dump it */ diff --git a/test/test_rewind.cxx b/test/test_rewind.cxx index 3ab37427a..882352a92 100644 --- a/test/test_rewind.cxx +++ b/test/test_rewind.cxx @@ -37,8 +37,7 @@ public: return remaining == 0; } - size_t Read(void *ptr, size_t read_size, - gcc_unused Error &error) override { + size_t Read(void *ptr, size_t read_size) override { size_t nbytes = std::min(remaining, read_size); memcpy(ptr, data, nbytes); data += nbytes; @@ -76,50 +75,50 @@ public: Error error; char buffer[16]; - size_t nbytes = ris->Read(buffer, 2, error); + size_t nbytes = ris->Read(buffer, 2); CPPUNIT_ASSERT_EQUAL(size_t(2), nbytes); CPPUNIT_ASSERT_EQUAL('f', buffer[0]); CPPUNIT_ASSERT_EQUAL('o', buffer[1]); CPPUNIT_ASSERT_EQUAL(offset_type(2), ris->GetOffset()); CPPUNIT_ASSERT(!ris->IsEOF()); - nbytes = ris->Read(buffer, 2, error); + nbytes = ris->Read(buffer, 2); CPPUNIT_ASSERT_EQUAL(size_t(2), nbytes); CPPUNIT_ASSERT_EQUAL('o', buffer[0]); CPPUNIT_ASSERT_EQUAL(' ', buffer[1]); CPPUNIT_ASSERT_EQUAL(offset_type(4), ris->GetOffset()); CPPUNIT_ASSERT(!ris->IsEOF()); - CPPUNIT_ASSERT(ris->Seek(1, error)); + ris->Seek(1); CPPUNIT_ASSERT_EQUAL(offset_type(1), ris->GetOffset()); CPPUNIT_ASSERT(!ris->IsEOF()); - nbytes = ris->Read(buffer, 2, error); + nbytes = ris->Read(buffer, 2); CPPUNIT_ASSERT_EQUAL(size_t(2), nbytes); CPPUNIT_ASSERT_EQUAL('o', buffer[0]); CPPUNIT_ASSERT_EQUAL('o', buffer[1]); CPPUNIT_ASSERT_EQUAL(offset_type(3), ris->GetOffset()); CPPUNIT_ASSERT(!ris->IsEOF()); - CPPUNIT_ASSERT(ris->Seek(0, error)); + ris->Seek(0); CPPUNIT_ASSERT_EQUAL(offset_type(0), ris->GetOffset()); CPPUNIT_ASSERT(!ris->IsEOF()); - nbytes = ris->Read(buffer, 2, error); + nbytes = ris->Read(buffer, 2); CPPUNIT_ASSERT_EQUAL(size_t(2), nbytes); CPPUNIT_ASSERT_EQUAL('f', buffer[0]); CPPUNIT_ASSERT_EQUAL('o', buffer[1]); CPPUNIT_ASSERT_EQUAL(offset_type(2), ris->GetOffset()); CPPUNIT_ASSERT(!ris->IsEOF()); - nbytes = ris->Read(buffer, sizeof(buffer), error); + nbytes = ris->Read(buffer, sizeof(buffer)); CPPUNIT_ASSERT_EQUAL(size_t(2), nbytes); CPPUNIT_ASSERT_EQUAL('o', buffer[0]); CPPUNIT_ASSERT_EQUAL(' ', buffer[1]); CPPUNIT_ASSERT_EQUAL(offset_type(4), ris->GetOffset()); CPPUNIT_ASSERT(!ris->IsEOF()); - nbytes = ris->Read(buffer, sizeof(buffer), error); + nbytes = ris->Read(buffer, sizeof(buffer)); CPPUNIT_ASSERT_EQUAL(size_t(3), nbytes); CPPUNIT_ASSERT_EQUAL('b', buffer[0]); CPPUNIT_ASSERT_EQUAL('a', buffer[1]); @@ -127,11 +126,11 @@ public: CPPUNIT_ASSERT_EQUAL(offset_type(7), ris->GetOffset()); CPPUNIT_ASSERT(ris->IsEOF()); - CPPUNIT_ASSERT(ris->Seek(3, error)); + ris->Seek(3); CPPUNIT_ASSERT_EQUAL(offset_type(3), ris->GetOffset()); CPPUNIT_ASSERT(!ris->IsEOF()); - nbytes = ris->Read(buffer, sizeof(buffer), error); + nbytes = ris->Read(buffer, sizeof(buffer)); CPPUNIT_ASSERT_EQUAL(size_t(4), nbytes); CPPUNIT_ASSERT_EQUAL(' ', buffer[0]); CPPUNIT_ASSERT_EQUAL('b', buffer[1]);