added another alert to cover all cases where the torrent has to be checked at start, and tell the user why. Fixed a problem with the resume data generation introduced in last storage check-in
This commit is contained in:
parent
4d19f7ff0f
commit
ffe2e75882
|
@ -187,7 +187,8 @@ namespace libtorrent
|
||||||
~piece_manager();
|
~piece_manager();
|
||||||
|
|
||||||
bool check_fastresume(aux::piece_checker_data& d
|
bool check_fastresume(aux::piece_checker_data& d
|
||||||
, std::vector<bool>& pieces, int& num_pieces, storage_mode_t storage_mode);
|
, std::vector<bool>& pieces, int& num_pieces, storage_mode_t storage_mode
|
||||||
|
, std::string& error_msg);
|
||||||
std::pair<bool, float> check_files(std::vector<bool>& pieces
|
std::pair<bool, float> check_files(std::vector<bool>& pieces
|
||||||
, int& num_pieces, boost::recursive_mutex& mutex);
|
, int& num_pieces, boost::recursive_mutex& mutex);
|
||||||
|
|
||||||
|
@ -236,7 +237,8 @@ namespace libtorrent
|
||||||
// slots to the piece that is stored (or
|
// slots to the piece that is stored (or
|
||||||
// partially stored) there. -2 is the index
|
// partially stored) there. -2 is the index
|
||||||
// of unassigned pieces and -1 is unallocated
|
// of unassigned pieces and -1 is unallocated
|
||||||
void export_piece_map(std::vector<int>& pieces) const;
|
void export_piece_map(std::vector<int>& pieces
|
||||||
|
, std::vector<bool> const& have) const;
|
||||||
|
|
||||||
bool compact_allocation() const
|
bool compact_allocation() const
|
||||||
{ return m_storage_mode == storage_mode_compact; }
|
{ return m_storage_mode == storage_mode_compact; }
|
||||||
|
|
|
@ -189,9 +189,11 @@ namespace detail
|
||||||
t->parse_resume_data(t->resume_data, t->torrent_ptr->torrent_file()
|
t->parse_resume_data(t->resume_data, t->torrent_ptr->torrent_file()
|
||||||
, error_msg);
|
, error_msg);
|
||||||
|
|
||||||
|
// lock the session to add the new torrent
|
||||||
|
session_impl::mutex_t::scoped_lock l(m_ses.m_mutex);
|
||||||
|
|
||||||
if (!error_msg.empty() && m_ses.m_alerts.should_post(alert::warning))
|
if (!error_msg.empty() && m_ses.m_alerts.should_post(alert::warning))
|
||||||
{
|
{
|
||||||
session_impl::mutex_t::scoped_lock l(m_ses.m_mutex);
|
|
||||||
m_ses.m_alerts.post_alert(fastresume_rejected_alert(
|
m_ses.m_alerts.post_alert(fastresume_rejected_alert(
|
||||||
t->torrent_ptr->get_handle()
|
t->torrent_ptr->get_handle()
|
||||||
, error_msg));
|
, error_msg));
|
||||||
|
@ -202,8 +204,6 @@ namespace detail
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
// lock the session to add the new torrent
|
|
||||||
session_impl::mutex_t::scoped_lock l(m_ses.m_mutex);
|
|
||||||
mutex::scoped_lock l2(m_mutex);
|
mutex::scoped_lock l2(m_mutex);
|
||||||
|
|
||||||
if (m_torrents.empty() || m_torrents.front() != t)
|
if (m_torrents.empty() || m_torrents.front() != t)
|
||||||
|
|
|
@ -1078,13 +1078,16 @@ namespace libtorrent
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
void piece_manager::export_piece_map(
|
void piece_manager::export_piece_map(
|
||||||
std::vector<int>& p) const
|
std::vector<int>& p, std::vector<bool> const& have) const
|
||||||
{
|
{
|
||||||
boost::recursive_mutex::scoped_lock lock(m_mutex);
|
boost::recursive_mutex::scoped_lock lock(m_mutex);
|
||||||
|
|
||||||
INVARIANT_CHECK;
|
INVARIANT_CHECK;
|
||||||
|
|
||||||
|
if (m_storage_mode == storage_mode_compact)
|
||||||
|
{
|
||||||
p.clear();
|
p.clear();
|
||||||
|
p.reserve(m_info->num_pieces());
|
||||||
std::vector<int>::const_reverse_iterator last;
|
std::vector<int>::const_reverse_iterator last;
|
||||||
for (last = m_slot_to_piece.rbegin();
|
for (last = m_slot_to_piece.rbegin();
|
||||||
last != m_slot_to_piece.rend(); ++last)
|
last != m_slot_to_piece.rend(); ++last)
|
||||||
|
@ -1099,6 +1102,15 @@ namespace libtorrent
|
||||||
p.push_back(*i);
|
p.push_back(*i);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
p.reserve(m_info->num_pieces());
|
||||||
|
for (int i = 0; i < m_info->num_pieces(); ++i)
|
||||||
|
{
|
||||||
|
p.push_back(have[i] ? i : unassigned);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void piece_manager::mark_failed(int piece_index)
|
void piece_manager::mark_failed(int piece_index)
|
||||||
{
|
{
|
||||||
|
@ -1374,7 +1386,8 @@ namespace libtorrent
|
||||||
bool piece_manager::check_fastresume(
|
bool piece_manager::check_fastresume(
|
||||||
aux::piece_checker_data& data
|
aux::piece_checker_data& data
|
||||||
, std::vector<bool>& pieces
|
, std::vector<bool>& pieces
|
||||||
, int& num_pieces, storage_mode_t storage_mode)
|
, int& num_pieces, storage_mode_t storage_mode
|
||||||
|
, std::string& error_msg)
|
||||||
{
|
{
|
||||||
boost::recursive_mutex::scoped_lock lock(m_mutex);
|
boost::recursive_mutex::scoped_lock lock(m_mutex);
|
||||||
|
|
||||||
|
@ -1473,6 +1486,7 @@ namespace libtorrent
|
||||||
// we're resuming a compact allocated storage
|
// we're resuming a compact allocated storage
|
||||||
m_state = state_expand_pieces;
|
m_state = state_expand_pieces;
|
||||||
m_current_slot = 0;
|
m_current_slot = 0;
|
||||||
|
error_msg = "pieces needs to be reordered";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1481,6 +1495,7 @@ namespace libtorrent
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
error_msg = "empty piece map";
|
||||||
m_state = state_full_check;
|
m_state = state_full_check;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -2215,10 +2215,22 @@ namespace libtorrent
|
||||||
bool done = true;
|
bool done = true;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
std::string error_msg;
|
||||||
TORRENT_ASSERT(m_storage);
|
TORRENT_ASSERT(m_storage);
|
||||||
TORRENT_ASSERT(m_owning_storage.get());
|
TORRENT_ASSERT(m_owning_storage.get());
|
||||||
done = m_storage->check_fastresume(data, m_have_pieces, m_num_pieces
|
done = m_storage->check_fastresume(data, m_have_pieces, m_num_pieces
|
||||||
, m_storage_mode);
|
, m_storage_mode, error_msg);
|
||||||
|
|
||||||
|
if (!error_msg.empty() && m_ses.m_alerts.should_post(alert::warning))
|
||||||
|
{
|
||||||
|
m_ses.m_alerts.post_alert(fastresume_rejected_alert(
|
||||||
|
get_handle(), error_msg));
|
||||||
|
#if defined(TORRENT_VERBOSE_LOGGING) || defined(TORRENT_LOGGING)
|
||||||
|
(*m_ses.m_logger) << "fastresume data for "
|
||||||
|
<< torrent_file().name() << " rejected: "
|
||||||
|
<< error_msg << "\n";
|
||||||
|
#endif
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch (std::exception& e)
|
catch (std::exception& e)
|
||||||
{
|
{
|
||||||
|
|
|
@ -661,7 +661,7 @@ namespace libtorrent
|
||||||
|
|
||||||
if (!t->valid_metadata()) return entry();
|
if (!t->valid_metadata()) return entry();
|
||||||
|
|
||||||
t->filesystem().export_piece_map(piece_index);
|
t->filesystem().export_piece_map(piece_index, t->pieces());
|
||||||
|
|
||||||
entry ret(entry::dictionary_t);
|
entry ret(entry::dictionary_t);
|
||||||
|
|
||||||
|
@ -673,10 +673,6 @@ namespace libtorrent
|
||||||
const sha1_hash& info_hash = t->torrent_file().info_hash();
|
const sha1_hash& info_hash = t->torrent_file().info_hash();
|
||||||
ret["info-hash"] = std::string((char*)info_hash.begin(), (char*)info_hash.end());
|
ret["info-hash"] = std::string((char*)info_hash.begin(), (char*)info_hash.end());
|
||||||
|
|
||||||
ret["slots"] = entry(entry::list_t);
|
|
||||||
entry::list_type& slots = ret["slots"].list();
|
|
||||||
std::copy(piece_index.begin(), piece_index.end(), std::back_inserter(slots));
|
|
||||||
|
|
||||||
// blocks per piece
|
// blocks per piece
|
||||||
int num_blocks_per_piece =
|
int num_blocks_per_piece =
|
||||||
static_cast<int>(t->torrent_file().piece_length()) / t->block_size();
|
static_cast<int>(t->torrent_file().piece_length()) / t->block_size();
|
||||||
|
@ -706,6 +702,13 @@ namespace libtorrent
|
||||||
// the unfinished piece's index
|
// the unfinished piece's index
|
||||||
piece_struct["piece"] = i->index;
|
piece_struct["piece"] = i->index;
|
||||||
|
|
||||||
|
// if the unfinished piece is not in the exported piece map, it means
|
||||||
|
// we're in full or sparse storage mode, in which case we have
|
||||||
|
// to insert the unfinished piece where it's stored
|
||||||
|
if (std::find(piece_index.begin(), piece_index.end(), i->index)
|
||||||
|
== piece_index.end() && i->index < int(piece_index.size()))
|
||||||
|
piece_index[i->index] = i->index;
|
||||||
|
|
||||||
std::string bitmask;
|
std::string bitmask;
|
||||||
const int num_bitmask_bytes
|
const int num_bitmask_bytes
|
||||||
= (std::max)(num_blocks_per_piece / 8, 1);
|
= (std::max)(num_blocks_per_piece / 8, 1);
|
||||||
|
@ -735,6 +738,10 @@ namespace libtorrent
|
||||||
up.push_back(piece_struct);
|
up.push_back(piece_struct);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
entry::list_type& slots = ret["slots"].list();
|
||||||
|
std::copy(piece_index.begin(), piece_index.end(), std::back_inserter(slots));
|
||||||
|
|
||||||
// write local peers
|
// write local peers
|
||||||
|
|
||||||
entry::list_type& peer_list = ret["peers"].list();
|
entry::list_type& peer_list = ret["peers"].list();
|
||||||
|
|
Loading…
Reference in New Issue