xz: Fix input file position when --single-stream is used.

Now the following works as you would expect:

    echo foo | xz > foo.xz
    echo bar | xz >> foo.xz
    ( xz -dc --single-stream ; xz -dc --single-stream ) < foo.xz

Note that it doesn't work if the input is not seekable
or if there is Stream Padding between the concatenated
.xz Streams.
This commit is contained in:
Lasse Collin 2011-05-01 12:24:23 +03:00
parent c29e6630c1
commit 7a480e4859
3 changed files with 29 additions and 0 deletions

View File

@ -594,6 +594,7 @@ coder_normal(file_pair *pair)
if (ret == LZMA_STREAM_END) { if (ret == LZMA_STREAM_END) {
if (opt_single_stream) { if (opt_single_stream) {
io_fix_src_pos(pair, strm.avail_in);
success = true; success = true;
break; break;
} }

View File

@ -840,6 +840,21 @@ io_close(file_pair *pair, bool success)
} }
extern void
io_fix_src_pos(file_pair *pair, size_t rewind_size)
{
assert(rewind_size <= IO_BUFFER_SIZE);
if (rewind_size > 0) {
// This doesn't need to work on unseekable file descriptors,
// so just ignore possible errors.
(void)lseek(pair->src_fd, -(off_t)(rewind_size), SEEK_CUR);
}
return;
}
extern size_t extern size_t
io_read(file_pair *pair, io_buf *buf_union, size_t size) io_read(file_pair *pair, io_buf *buf_union, size_t size)
{ {

View File

@ -102,6 +102,19 @@ extern void io_close(file_pair *pair, bool success);
extern size_t io_read(file_pair *pair, io_buf *buf, size_t size); extern size_t io_read(file_pair *pair, io_buf *buf, size_t size);
/// \brief Fix the position in src_fd
///
/// This is used when --single-thream has been specified and decompression
/// is successful. If the input file descriptor supports seeking, this
/// function fixes the input position to point to the next byte after the
/// decompressed stream.
///
/// \param pair File pair having the source file open for reading
/// \param rewind_size How many bytes of extra have been read i.e.
/// how much to seek backwards.
extern void io_fix_src_pos(file_pair *pair, size_t rewind_size);
/// \brief Read from source file from given offset to a buffer /// \brief Read from source file from given offset to a buffer
/// ///
/// This is remotely similar to standard pread(). This uses lseek() though, /// This is remotely similar to standard pread(). This uses lseek() though,