fixed another sparse files related bug in storage

This commit is contained in:
Arvid Norberg 2007-05-04 03:28:42 +00:00
parent f942308ccb
commit c39462c125
1 changed files with 78 additions and 73 deletions

View File

@ -628,95 +628,100 @@ namespace libtorrent
++file_iter; ++file_iter;
} }
boost::shared_ptr<file> in(m_files.open_file(
this, m_save_path / file_iter->path, file::in));
assert(file_offset < file_iter->size);
assert(slices[0].offset == file_offset);
size_type new_pos = in->seek(file_offset);
if (new_pos != file_offset)
{
// the file was not big enough
throw file_error("slot has no storage");
}
#ifndef NDEBUG
size_type in_tell = in->tell();
assert(in_tell == file_offset);
#endif
int left_to_read = size;
int slot_size = static_cast<int>(m_info.piece_size(slot));
if (offset + left_to_read > slot_size)
left_to_read = slot_size - offset;
assert(left_to_read >= 0);
size_type result = left_to_read;
int buf_pos = 0; int buf_pos = 0;
try
#ifndef NDEBUG
int counter = 0;
#endif
while (left_to_read > 0)
{ {
int read_bytes = left_to_read; boost::shared_ptr<file> in(m_files.open_file(
if (file_offset + read_bytes > file_iter->size) this, m_save_path / file_iter->path, file::in));
read_bytes = static_cast<int>(file_iter->size - file_offset);
if (read_bytes > 0) assert(file_offset < file_iter->size);
assert(slices[0].offset == file_offset);
size_type new_pos = in->seek(file_offset);
if (new_pos != file_offset)
{ {
#ifndef NDEBUG // the file was not big enough
assert(int(slices.size()) > counter); throw file_error("slot has no storage");
size_type slice_size = slices[counter].size; }
assert(slice_size == read_bytes);
assert(m_info.file_at(slices[counter].file_index).path
== file_iter->path);
#endif
size_type actual_read = in->read(buf + buf_pos, read_bytes); #ifndef NDEBUG
size_type in_tell = in->tell();
assert(in_tell == file_offset);
#endif
if (read_bytes != actual_read) int left_to_read = size;
int slot_size = static_cast<int>(m_info.piece_size(slot));
if (offset + left_to_read > slot_size)
left_to_read = slot_size - offset;
assert(left_to_read >= 0);
size_type result = left_to_read;
#ifndef NDEBUG
int counter = 0;
#endif
while (left_to_read > 0)
{
int read_bytes = left_to_read;
if (file_offset + read_bytes > file_iter->size)
read_bytes = static_cast<int>(file_iter->size - file_offset);
if (read_bytes > 0)
{ {
if (fill_zero && actual_read < read_bytes) #ifndef NDEBUG
{ assert(int(slices.size()) > counter);
std::memset(buf + buf_pos + actual_read, 0, read_bytes - actual_read); size_type slice_size = slices[counter].size;
} assert(slice_size == read_bytes);
else assert(m_info.file_at(slices[counter].file_index).path
== file_iter->path);
#endif
size_type actual_read = in->read(buf + buf_pos, read_bytes);
if (read_bytes != actual_read)
{ {
if (actual_read > 0) buf_pos += actual_read;
// the file was not big enough // the file was not big enough
throw file_error("slot has no storage"); throw file_error("slot has no storage");
} }
left_to_read -= read_bytes;
buf_pos += read_bytes;
assert(buf_pos >= 0);
file_offset += read_bytes;
} }
left_to_read -= read_bytes; if (left_to_read > 0)
buf_pos += read_bytes; {
assert(buf_pos >= 0); ++file_iter;
file_offset += read_bytes; #ifndef NDEBUG
} // empty files are not returned by map_block, so if
// this file was empty, don't increment the slice counter
if (read_bytes > 0) ++counter;
#endif
path path = m_save_path / file_iter->path;
if (left_to_read > 0) file_offset = 0;
{ in = m_files.open_file(
++file_iter; this, path, file::in);
#ifndef NDEBUG in->seek(0);
// empty files are not returned by map_block, so if }
// this file was empty, don't increment the slice counter
if (read_bytes > 0) ++counter;
#endif
path path = m_save_path / file_iter->path;
file_offset = 0;
in = m_files.open_file(
this, path, file::in);
in->seek(0);
} }
return result;
}
catch (std::exception&)
{
if (fill_zero)
{
std::memset(buf + buf_pos, 0, size - buf_pos);
return size;
}
throw;
} }
return result;
} }
// throws file_error if it fails to write // throws file_error if it fails to write