Merge branch 'v0.21.x' into master
This commit is contained in:
commit
e8380cf2aa
4
NEWS
4
NEWS
@ -13,8 +13,6 @@ ver 0.22 (not yet released)
|
|||||||
- ffmpeg: allow partial reads
|
- ffmpeg: allow partial reads
|
||||||
- io_uring: new plugin for local files on Linux (using liburing)
|
- io_uring: new plugin for local files on Linux (using liburing)
|
||||||
- smbclient: close unused SMB/CIFS connections
|
- smbclient: close unused SMB/CIFS connections
|
||||||
* archive
|
|
||||||
- iso9660: support seeking
|
|
||||||
* database
|
* database
|
||||||
- upnp: drop support for libupnp versions older than 1.8
|
- upnp: drop support for libupnp versions older than 1.8
|
||||||
* playlist
|
* playlist
|
||||||
@ -48,6 +46,8 @@ ver 0.21.26 (not yet released)
|
|||||||
* archive
|
* archive
|
||||||
- bzip2: fix crash on corrupt bzip2 file
|
- bzip2: fix crash on corrupt bzip2 file
|
||||||
- bzip2: flush output at end of input file
|
- bzip2: flush output at end of input file
|
||||||
|
- iso9660: fix unaligned reads
|
||||||
|
- iso9660: support seeking
|
||||||
- zzip: fix crash on corrupt ZIP file
|
- zzip: fix crash on corrupt ZIP file
|
||||||
* decoder
|
* decoder
|
||||||
- sndfile: fix lost samples at end of file
|
- sndfile: fix lost samples at end of file
|
||||||
|
@ -29,9 +29,12 @@
|
|||||||
#include "fs/Path.hxx"
|
#include "fs/Path.hxx"
|
||||||
#include "util/RuntimeError.hxx"
|
#include "util/RuntimeError.hxx"
|
||||||
#include "util/StringCompare.hxx"
|
#include "util/StringCompare.hxx"
|
||||||
|
#include "util/WritableBuffer.hxx"
|
||||||
|
|
||||||
#include <cdio/iso9660.h>
|
#include <cdio/iso9660.h>
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
@ -150,6 +153,47 @@ class Iso9660InputStream final : public InputStream {
|
|||||||
|
|
||||||
const lsn_t lsn;
|
const lsn_t lsn;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* libiso9660 can only read whole sectors at a time, and this
|
||||||
|
* buffer is used to store one whole sector and allow Read()
|
||||||
|
* to handle partial sector reads.
|
||||||
|
*/
|
||||||
|
class BlockBuffer {
|
||||||
|
size_t position = 0, fill = 0;
|
||||||
|
|
||||||
|
std::array<uint8_t, ISO_BLOCKSIZE> data;
|
||||||
|
|
||||||
|
public:
|
||||||
|
ConstBuffer<uint8_t> Read() const noexcept {
|
||||||
|
assert(fill <= data.size());
|
||||||
|
assert(position <= fill);
|
||||||
|
|
||||||
|
return {&data[position], &data[fill]};
|
||||||
|
}
|
||||||
|
|
||||||
|
void Consume(size_t nbytes) noexcept {
|
||||||
|
assert(nbytes <= Read().size);
|
||||||
|
|
||||||
|
position += nbytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
WritableBuffer<uint8_t> Write() noexcept {
|
||||||
|
assert(Read().empty());
|
||||||
|
|
||||||
|
return {data.data(), data.size()};
|
||||||
|
}
|
||||||
|
|
||||||
|
void Append(size_t nbytes) noexcept {
|
||||||
|
assert(Read().empty());
|
||||||
|
assert(nbytes <= data.size());
|
||||||
|
|
||||||
|
fill = nbytes;
|
||||||
|
position = 0;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
BlockBuffer buffer;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
Iso9660InputStream(std::shared_ptr<Iso9660> _iso,
|
Iso9660InputStream(std::shared_ptr<Iso9660> _iso,
|
||||||
const char *_uri,
|
const char *_uri,
|
||||||
@ -170,6 +214,9 @@ public:
|
|||||||
void *ptr, size_t size) override;
|
void *ptr, size_t size) override;
|
||||||
|
|
||||||
void Seek(std::unique_lock<Mutex> &, offset_type new_offset) override {
|
void Seek(std::unique_lock<Mutex> &, offset_type new_offset) override {
|
||||||
|
if (new_offset > size)
|
||||||
|
throw std::runtime_error("Invalid seek offset");
|
||||||
|
|
||||||
offset = new_offset;
|
offset = new_offset;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -195,35 +242,56 @@ size_t
|
|||||||
Iso9660InputStream::Read(std::unique_lock<Mutex> &,
|
Iso9660InputStream::Read(std::unique_lock<Mutex> &,
|
||||||
void *ptr, size_t read_size)
|
void *ptr, size_t read_size)
|
||||||
{
|
{
|
||||||
const ScopeUnlock unlock(mutex);
|
const offset_type remaining = size - offset;
|
||||||
|
if (remaining == 0)
|
||||||
int readed = 0;
|
|
||||||
int no_blocks, cur_block;
|
|
||||||
size_t left_bytes = size - offset;
|
|
||||||
|
|
||||||
if (left_bytes < read_size) {
|
|
||||||
no_blocks = CEILING(left_bytes, ISO_BLOCKSIZE);
|
|
||||||
} else {
|
|
||||||
no_blocks = read_size / ISO_BLOCKSIZE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (no_blocks == 0)
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
cur_block = offset / ISO_BLOCKSIZE;
|
if (offset_type(read_size) > remaining)
|
||||||
|
read_size = remaining;
|
||||||
|
|
||||||
readed = iso->SeekRead(ptr, lsn + cur_block, no_blocks);
|
auto r = buffer.Read();
|
||||||
|
|
||||||
if (readed != no_blocks * ISO_BLOCKSIZE)
|
if (r.empty()) {
|
||||||
throw FormatRuntimeError("error reading ISO file at lsn %lu",
|
/* the buffer is empty - read more data from the ISO file */
|
||||||
(unsigned long)cur_block);
|
|
||||||
|
|
||||||
if (left_bytes < read_size) {
|
assert(offset % ISO_BLOCKSIZE == 0);
|
||||||
readed = left_bytes;
|
|
||||||
|
const ScopeUnlock unlock(mutex);
|
||||||
|
|
||||||
|
const lsn_t read_lsn = lsn + offset / ISO_BLOCKSIZE;
|
||||||
|
|
||||||
|
if (read_size >= ISO_BLOCKSIZE) {
|
||||||
|
/* big read - read right into the caller's buffer */
|
||||||
|
|
||||||
|
auto nbytes = iso->SeekRead(ptr, read_lsn,
|
||||||
|
read_size / ISO_BLOCKSIZE);
|
||||||
|
if (nbytes <= 0)
|
||||||
|
throw std::runtime_error("Failed to read ISO9660 file");
|
||||||
|
|
||||||
|
offset += nbytes;
|
||||||
|
return nbytes;
|
||||||
}
|
}
|
||||||
|
|
||||||
offset += readed;
|
/* fill the buffer */
|
||||||
return readed;
|
|
||||||
|
auto w = buffer.Write();
|
||||||
|
auto nbytes = iso->SeekRead(w.data, read_lsn,
|
||||||
|
w.size / ISO_BLOCKSIZE);
|
||||||
|
if (nbytes <= 0)
|
||||||
|
throw std::runtime_error("Failed to read ISO9660 file");
|
||||||
|
|
||||||
|
buffer.Append(nbytes);
|
||||||
|
|
||||||
|
r = buffer.Read();
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(!r.empty());
|
||||||
|
|
||||||
|
size_t nbytes = std::min(read_size, r.size);
|
||||||
|
memcpy(ptr, r.data, nbytes);
|
||||||
|
buffer.Consume(nbytes);
|
||||||
|
offset += nbytes;
|
||||||
|
return nbytes;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
|
@ -49,11 +49,15 @@
|
|||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
static constexpr std::size_t MAX_CHUNK_SIZE = 16384;
|
||||||
|
|
||||||
struct CommandLine {
|
struct CommandLine {
|
||||||
const char *uri = nullptr;
|
const char *uri = nullptr;
|
||||||
|
|
||||||
FromNarrowPath config_path;
|
FromNarrowPath config_path;
|
||||||
|
|
||||||
|
std::size_t chunk_size = MAX_CHUNK_SIZE;
|
||||||
|
|
||||||
bool verbose = false;
|
bool verbose = false;
|
||||||
|
|
||||||
bool scan = false;
|
bool scan = false;
|
||||||
@ -63,14 +67,27 @@ enum Option {
|
|||||||
OPTION_CONFIG,
|
OPTION_CONFIG,
|
||||||
OPTION_VERBOSE,
|
OPTION_VERBOSE,
|
||||||
OPTION_SCAN,
|
OPTION_SCAN,
|
||||||
|
OPTION_CHUNK_SIZE,
|
||||||
};
|
};
|
||||||
|
|
||||||
static constexpr OptionDef option_defs[] = {
|
static constexpr OptionDef option_defs[] = {
|
||||||
{"config", 0, true, "Load a MPD configuration file"},
|
{"config", 0, true, "Load a MPD configuration file"},
|
||||||
{"verbose", 'v', false, "Verbose logging"},
|
{"verbose", 'v', false, "Verbose logging"},
|
||||||
{"scan", 0, false, "Scan tags instead of reading raw data"},
|
{"scan", 0, false, "Scan tags instead of reading raw data"},
|
||||||
|
{"chunk-size", 0, true, "Read this number of bytes at a time"},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static std::size_t
|
||||||
|
ParseSize(const char *s)
|
||||||
|
{
|
||||||
|
char *endptr;
|
||||||
|
std::size_t value = std::strtoul(s, &endptr, 10);
|
||||||
|
if (endptr == s)
|
||||||
|
throw std::runtime_error("Failed to parse integer");
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
static CommandLine
|
static CommandLine
|
||||||
ParseCommandLine(int argc, char **argv)
|
ParseCommandLine(int argc, char **argv)
|
||||||
{
|
{
|
||||||
@ -90,6 +107,12 @@ ParseCommandLine(int argc, char **argv)
|
|||||||
case OPTION_SCAN:
|
case OPTION_SCAN:
|
||||||
c.scan = true;
|
c.scan = true;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case OPTION_CHUNK_SIZE:
|
||||||
|
c.chunk_size = ParseSize(o.value);
|
||||||
|
if (c.chunk_size <= 0 || c.chunk_size > MAX_CHUNK_SIZE)
|
||||||
|
throw std::runtime_error("Invalid chunk size");
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -130,7 +153,7 @@ tag_save(FILE *file, const Tag &tag)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
dump_input_stream(InputStream &is, FileDescriptor out)
|
dump_input_stream(InputStream &is, FileDescriptor out, size_t chunk_size)
|
||||||
{
|
{
|
||||||
std::unique_lock<Mutex> lock(is.mutex);
|
std::unique_lock<Mutex> lock(is.mutex);
|
||||||
|
|
||||||
@ -150,8 +173,9 @@ dump_input_stream(InputStream &is, FileDescriptor out)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
char buffer[4096];
|
char buffer[MAX_CHUNK_SIZE];
|
||||||
size_t num_read = is.Read(lock, buffer, sizeof(buffer));
|
assert(chunk_size <= sizeof(buffer));
|
||||||
|
size_t num_read = is.Read(lock, buffer, chunk_size);
|
||||||
if (num_read == 0)
|
if (num_read == 0)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -231,7 +255,8 @@ try {
|
|||||||
|
|
||||||
Mutex mutex;
|
Mutex mutex;
|
||||||
auto is = InputStream::OpenReady(c.uri, mutex);
|
auto is = InputStream::OpenReady(c.uri, mutex);
|
||||||
return dump_input_stream(*is, FileDescriptor(STDOUT_FILENO));
|
return dump_input_stream(*is, FileDescriptor(STDOUT_FILENO),
|
||||||
|
c.chunk_size);
|
||||||
} catch (...) {
|
} catch (...) {
|
||||||
PrintException(std::current_exception());
|
PrintException(std::current_exception());
|
||||||
return EXIT_FAILURE;
|
return EXIT_FAILURE;
|
||||||
|
@ -7,4 +7,7 @@ DST="$(pwd)/test/tmp/${SRC_BASE}.iso"
|
|||||||
mkdir -p test/tmp
|
mkdir -p test/tmp
|
||||||
rm -f "$DST"
|
rm -f "$DST"
|
||||||
mkisofs -quiet -l -o "$DST" "$SRC"
|
mkisofs -quiet -l -o "$DST" "$SRC"
|
||||||
./test/run_input "$DST/${SRC_BASE}" |diff "$SRC" -
|
|
||||||
|
# Using an odd chunk size to check whether the plugin can cope with
|
||||||
|
# partial sectors
|
||||||
|
./test/run_input --chunk-size=1337 "$DST/${SRC_BASE}" |diff "$SRC" -
|
||||||
|
Loading…
Reference in New Issue
Block a user