|
|
b5048e |
From b451bfd224da44e93cf842f23455d755e48421dd Mon Sep 17 00:00:00 2001
|
|
|
b5048e |
From: Pavel Raiskup <praiskup@redhat.com>
|
|
|
b5048e |
Date: Mon, 28 Jul 2014 08:17:55 +0200
|
|
|
b5048e |
Subject: [PATCH 8/9] Fix for infinite loops during sparse file handling
|
|
|
b5048e |
|
|
|
b5048e |
Upstream bugreport (still downstream):
|
|
|
b5048e |
http://www.mail-archive.com/bug-tar@gnu.org/msg04432.html
|
|
|
b5048e |
|
|
|
b5048e |
Resolves: #1082608
|
|
|
b5048e |
|
|
|
b5048e |
---
|
|
|
b5048e |
THANKS | 1 +
|
|
|
b5048e |
src/sparse.c | 48 ++++++++++++++++++++++++++++++++----------------
|
|
|
b5048e |
2 files changed, 33 insertions(+), 16 deletions(-)
|
|
|
b5048e |
|
|
|
b5048e |
diff --git a/THANKS b/THANKS
|
|
|
b5048e |
index b4c5427..e74f71c 100644
|
|
|
b5048e |
--- a/THANKS
|
|
|
b5048e |
+++ b/THANKS
|
|
|
b5048e |
@@ -175,6 +175,7 @@ Fabio d'Alessi cars@civ.bio.unipd.it
|
|
|
b5048e |
Frank Heckenbach frank@g-n-u.de
|
|
|
b5048e |
Frank Koenen koenfr@lidp.com
|
|
|
b5048e |
Franz-Werner Gergen gergen@edvulx.mpi-stuttgart.mpg.de
|
|
|
b5048e |
+François Ouellet fouell@gmail.com
|
|
|
b5048e |
François Pinard pinard@iro.umontreal.ca
|
|
|
b5048e |
Fritz Elfert fritz@fsun.triltsch.de
|
|
|
b5048e |
George Chyu gschyu@ccgate.dp.beckman.com
|
|
|
b5048e |
diff --git a/src/sparse.c b/src/sparse.c
|
|
|
b5048e |
index 6a97676..53c1868 100644
|
|
|
b5048e |
--- a/src/sparse.c
|
|
|
b5048e |
+++ b/src/sparse.c
|
|
|
b5048e |
@@ -301,6 +301,7 @@ sparse_dump_region (struct tar_sparse_file *file, size_t i)
|
|
|
b5048e |
{
|
|
|
b5048e |
union block *blk;
|
|
|
b5048e |
off_t bytes_left = file->stat_info->sparse_map[i].numbytes;
|
|
|
b5048e |
+ const char *file_name = file->stat_info->orig_file_name;
|
|
|
b5048e |
|
|
|
b5048e |
if (!lseek_or_error (file, file->stat_info->sparse_map[i].offset))
|
|
|
b5048e |
return false;
|
|
|
b5048e |
@@ -314,13 +315,23 @@ sparse_dump_region (struct tar_sparse_file *file, size_t i)
|
|
|
b5048e |
bytes_read = safe_read (file->fd, blk->buffer, bufsize);
|
|
|
b5048e |
if (bytes_read == SAFE_READ_ERROR)
|
|
|
b5048e |
{
|
|
|
b5048e |
- read_diag_details (file->stat_info->orig_file_name,
|
|
|
b5048e |
+ read_diag_details (file_name,
|
|
|
b5048e |
(file->stat_info->sparse_map[i].offset
|
|
|
b5048e |
+ file->stat_info->sparse_map[i].numbytes
|
|
|
b5048e |
- bytes_left),
|
|
|
b5048e |
bufsize);
|
|
|
b5048e |
return false;
|
|
|
b5048e |
}
|
|
|
b5048e |
+ if (bytes_read == 0)
|
|
|
b5048e |
+ {
|
|
|
b5048e |
+ char buf[UINTMAX_STRSIZE_BOUND];
|
|
|
b5048e |
+ FATAL_ERROR ((0, 0,
|
|
|
b5048e |
+ ngettext ("%s: File shrank by %s byte",
|
|
|
b5048e |
+ "%s: File shrank by %s bytes",
|
|
|
b5048e |
+ bytes_left),
|
|
|
b5048e |
+ quotearg_colon (file_name),
|
|
|
b5048e |
+ offtostr (bytes_left, buf)));
|
|
|
b5048e |
+ }
|
|
|
b5048e |
|
|
|
b5048e |
memset (blk->buffer + bytes_read, 0, BLOCKSIZE - bytes_read);
|
|
|
b5048e |
bytes_left -= bytes_read;
|
|
|
b5048e |
@@ -475,33 +486,37 @@ sparse_skip_file (struct tar_stat_info *st)
|
|
|
b5048e |
static bool
|
|
|
b5048e |
check_sparse_region (struct tar_sparse_file *file, off_t beg, off_t end)
|
|
|
b5048e |
{
|
|
|
b5048e |
- if (!lseek_or_error (file, beg))
|
|
|
b5048e |
+ off_t offset = beg;
|
|
|
b5048e |
+
|
|
|
b5048e |
+ if (!lseek_or_error (file, offset))
|
|
|
b5048e |
return false;
|
|
|
b5048e |
|
|
|
b5048e |
- while (beg < end)
|
|
|
b5048e |
+ while (offset < end)
|
|
|
b5048e |
{
|
|
|
b5048e |
size_t bytes_read;
|
|
|
b5048e |
- size_t rdsize = BLOCKSIZE < end - beg ? BLOCKSIZE : end - beg;
|
|
|
b5048e |
+ size_t rdsize = BLOCKSIZE < end - offset ? BLOCKSIZE : end - offset;
|
|
|
b5048e |
char diff_buffer[BLOCKSIZE];
|
|
|
b5048e |
|
|
|
b5048e |
bytes_read = safe_read (file->fd, diff_buffer, rdsize);
|
|
|
b5048e |
if (bytes_read == SAFE_READ_ERROR)
|
|
|
b5048e |
{
|
|
|
b5048e |
read_diag_details (file->stat_info->orig_file_name,
|
|
|
b5048e |
- beg,
|
|
|
b5048e |
- rdsize);
|
|
|
b5048e |
- return false;
|
|
|
b5048e |
- }
|
|
|
b5048e |
- if (!zero_block_p (diff_buffer, bytes_read))
|
|
|
b5048e |
- {
|
|
|
b5048e |
- char begbuf[INT_BUFSIZE_BOUND (off_t)];
|
|
|
b5048e |
- report_difference (file->stat_info,
|
|
|
b5048e |
- _("File fragment at %s is not a hole"),
|
|
|
b5048e |
- offtostr (beg, begbuf));
|
|
|
b5048e |
+ offset, rdsize);
|
|
|
b5048e |
return false;
|
|
|
b5048e |
}
|
|
|
b5048e |
|
|
|
b5048e |
- beg += bytes_read;
|
|
|
b5048e |
+ if (bytes_read == 0
|
|
|
b5048e |
+ || !zero_block_p (diff_buffer, bytes_read))
|
|
|
b5048e |
+ {
|
|
|
b5048e |
+ char begbuf[INT_BUFSIZE_BOUND (off_t)];
|
|
|
b5048e |
+ const char *msg = bytes_read ? _("File fragment at %s is not a hole")
|
|
|
b5048e |
+ : _("Hole starting at %s is truncated");
|
|
|
b5048e |
+
|
|
|
b5048e |
+ report_difference (file->stat_info, msg, offtostr (beg, begbuf));
|
|
|
b5048e |
+ return false;
|
|
|
b5048e |
+ }
|
|
|
b5048e |
+
|
|
|
b5048e |
+ offset += bytes_read;
|
|
|
b5048e |
}
|
|
|
b5048e |
return true;
|
|
|
b5048e |
}
|
|
|
b5048e |
@@ -542,7 +557,8 @@ check_data_region (struct tar_sparse_file *file, size_t i)
|
|
|
b5048e |
file->dumped_size += bytes_read;
|
|
|
b5048e |
size_left -= bytes_read;
|
|
|
b5048e |
mv_size_left (file->stat_info->archive_file_size - file->dumped_size);
|
|
|
b5048e |
- if (memcmp (blk->buffer, diff_buffer, rdsize))
|
|
|
b5048e |
+ if (bytes_read == 0
|
|
|
b5048e |
+ || memcmp (blk->buffer, diff_buffer, bytes_read))
|
|
|
b5048e |
{
|
|
|
b5048e |
report_difference (file->stat_info, _("Contents differ"));
|
|
|
b5048e |
return false;
|
|
|
b5048e |
--
|
|
|
b5048e |
1.9.3
|
|
|
b5048e |
|