📄 download_wrapper.cc
字号:
// libTorrent - BitTorrent library// Copyright (C) 2005-2006, Jari Sundell//// This program is free software; you can redistribute it and/or modify// it under the terms of the GNU General Public License as published by// the Free Software Foundation; either version 2 of the License, or// (at your option) any later version.// // This program is distributed in the hope that it will be useful,// but WITHOUT ANY WARRANTY; without even the implied warranty of// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the// GNU General Public License for more details.// // You should have received a copy of the GNU General Public License// along with this program; if not, write to the Free Software// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA//// In addition, as a special exception, the copyright holders give// permission to link the code of portions of this program with the// OpenSSL library under certain conditions as described in each// individual source file, and distribute linked combinations// including the two.//// You must obey the GNU General Public License in all respects for// all of the code used other than OpenSSL. If you modify file(s)// with this exception, you may extend this exception to your version// of the file(s), but you are not obligated to do so. If you do not// wish to do so, delete this exception statement from your version.// If you delete this exception statement from all source files in the// program, then also delete it here.//// Contact: Jari Sundell <jaris@ifi.uio.no>//// Skomakerveien 33// 3185 Skoppum, NORWAY#include "config.h"#include <iterator>#include <stdlib.h>#include <rak/file_stat.h>#include <sigc++/bind.h>#include "data/chunk_list.h"#include "data/content.h"#include "data/hash_queue.h"#include "data/hash_torrent.h"#include "data/file_manager.h"#include "data/file_meta.h"#include "protocol/handshake_manager.h"#include "protocol/peer_connection_base.h"#include "torrent/exceptions.h"#include "torrent/object.h"#include "tracker/tracker_manager.h"#include "available_list.h"#include "chunk_selector.h"#include "download_wrapper.h"namespace torrent {DownloadWrapper::DownloadWrapper() : m_bencode(NULL), m_hash(NULL), m_connectionType(0) { m_delayDownloadDone.set_slot(rak::mem_fn(&m_signalDownloadDone, &Signal::operator())); m_main.tracker_manager()->set_info(info()); m_main.tracker_manager()->slot_success(rak::make_mem_fun(this, &DownloadWrapper::receive_tracker_success)); m_main.tracker_manager()->slot_failed(rak::make_mem_fun(this, &DownloadWrapper::receive_tracker_failed)); m_main.chunk_list()->slot_storage_error(rak::make_mem_fun(this, &DownloadWrapper::receive_storage_error));}DownloadWrapper::~DownloadWrapper() { if (info()->is_active()) m_main.stop(); if (info()->is_open()) close(); delete m_hash; delete m_bencode;}voidDownloadWrapper::initialize(const std::string& hash, const std::string& id) { m_main.slot_hash_check_add(rak::make_mem_fun(this, &DownloadWrapper::check_chunk_hash)); info()->set_hash(hash); info()->set_local_id(id); info()->slot_completed() = rak::make_mem_fun(m_main.content(), &Content::bytes_completed); info()->slot_left() = rak::make_mem_fun(m_main.content(), &Content::bytes_left); m_main.connection_list()->slot_connected(rak::make_mem_fun(this, &DownloadWrapper::receive_peer_connected)); m_main.connection_list()->slot_disconnected(rak::make_mem_fun(this, &DownloadWrapper::receive_peer_disconnected)); // Info hash must be calculate from here on. m_hash = new HashTorrent(m_main.chunk_list()); // Connect various signals and slots. m_hash->slot_check_chunk(rak::make_mem_fun(this, &DownloadWrapper::check_chunk_hash)); m_hash->slot_storage_error(rak::make_mem_fun(this, &DownloadWrapper::receive_storage_error)); m_hash->delay_checked().set_slot(rak::mem_fn(this, &DownloadWrapper::receive_initial_hash));}voidDownloadWrapper::open() { if (info()->is_open()) return; m_main.open(); m_hash->ranges().insert(0, m_main.content()->chunk_total());}voidDownloadWrapper::close() { // Stop the hashing first as we need to make sure all chunks are // released when DownloadMain::close() is called. m_hash->clear(); // Clear after m_hash to ensure that the empty hash done signal does // not get passed to HashTorrent. m_hash->get_queue()->remove(this); // This could/should be async as we do not care that much if it // succeeds or not, any chunks not included in that last // hash_resume_save get ignored anyway. m_main.chunk_list()->sync_chunks(ChunkList::sync_all | ChunkList::sync_force | ChunkList::sync_sloppy | ChunkList::sync_ignore_error); m_main.close(); // Should this perhaps be in stop? priority_queue_erase(&taskScheduler, &m_delayDownloadDone);}boolDownloadWrapper::is_stopped() const { return !m_main.tracker_manager()->is_active() && !m_main.tracker_manager()->is_busy();}voidDownloadWrapper::receive_initial_hash() { if (info()->is_active()) throw internal_error("DownloadWrapper::receive_initial_hash() but we're in a bad state."); if (!m_hash->is_checking()) { if (rak::error_number(m_hash->error_number()).is_valid()) info()->signal_storage_error().emit(("Hash checker was unable to map chunk: " + std::string(rak::error_number(m_hash->error_number()).c_str()))); m_hash->clear(); m_hash->ranges().clear(); // Clear after m_hash to ensure that the empty hash done signal does // not get passed to HashTorrent. m_hash->get_queue()->remove(this); m_main.content()->bitfield()->unallocate(); } else if (m_main.content()->entry_list()->resize_all()) { m_hash->confirm_checked(); if (m_hash->get_queue()->has(this)) throw internal_error("DownloadWrapper::receive_initial_hash() found a chunk in the HashQueue."); // Initialize the ChunkSelector here so that no chunks will be // marked by HashTorrent that are not accounted for. m_main.chunk_selector()->initialize(m_main.content()->bitfield(), m_main.chunk_statistics()); receive_update_priorities(); } else { // We couldn't resize the files, tell the client. receive_storage_error("Could not resize files in the torrent."); // Do we clear the hash? } m_signalInitialHash.emit();} voidDownloadWrapper::check_chunk_hash(ChunkHandle handle) { // Using HashTorrent's queue temporarily. m_hash->get_queue()->push_back(handle, rak::make_mem_fun(this, &DownloadWrapper::receive_hash_done));}voidDownloadWrapper::receive_hash_done(ChunkHandle handle, const char* hash) { if (!handle.is_valid()) throw internal_error("DownloadWrapper::receive_hash_done(...) called on an invalid chunk."); if (!info()->is_open()) throw internal_error("DownloadWrapper::receive_hash_done(...) called but the download is not open."); if (hash == NULL) { // Clearing the hash queue, do nothing. Add checks here? } else if (m_hash->is_checking()) { m_main.content()->receive_chunk_hash(handle.index(), hash); m_hash->receive_chunkdone(); } else if (m_hash->is_checked()) { // Receiving chunk hashes after stopping the torrent should be // safe. if (m_main.chunk_selector()->bitfield()->get(handle.index())) throw internal_error("DownloadWrapper::receive_hash_done(...) received a chunk that isn't set in ChunkSelector."); if (m_main.content()->receive_chunk_hash(handle.index(), hash)) { m_main.delegator()->transfer_list()->hash_succeded(handle.index()); m_main.update_endgame(); if (m_main.content()->is_done()) finished_download(); m_main.connection_list()->send_finished_chunk(handle.index()); signal_chunk_passed().emit(handle.index()); } else { // This needs to ensure the chunk is still valid. m_main.delegator()->transfer_list()->hash_failed(handle.index(), handle.chunk()); signal_chunk_failed().emit(handle.index()); } } else { // When clearing, we get a NULL string, so if none of the above // caught this hash, then we got a problem. throw internal_error("DownloadWrapper::receive_hash_done(...) Was not expecting non-NULL hash."); } m_main.chunk_list()->release(&handle);} voidDownloadWrapper::receive_storage_error(const std::string& str) {// m_main.chunk_list()->slot_storage_error(ChunkList::SlotStorageError(NULL, NULL)); m_main.stop(); m_main.close();// m_main.chunk_list()->slot_storage_error(rak::make_mem_fun(this, &DownloadWrapper::receive_storage_error)); m_main.tracker_manager()->set_active(false); m_main.tracker_manager()->close(); info()->signal_storage_error().emit(str);}voidDownloadWrapper::receive_tracker_success(AddressList* l) { m_main.connection_list()->set_difference(l); m_main.peer_list()->available_list()->insert(l); m_main.receive_connect_peers(); m_main.receive_tracker_success(); m_signalTrackerSuccess.emit();}voidDownloadWrapper::receive_tracker_failed(const std::string& msg) { m_signalTrackerFailed.emit(msg);}voidDownloadWrapper::receive_peer_connected(PeerConnectionBase* peer) { m_signalPeerConnected.emit(Peer(peer));}voidDownloadWrapper::receive_peer_disconnected(PeerConnectionBase* peer) { m_signalPeerDisconnected.emit(Peer(peer));}voidDownloadWrapper::receive_tick(uint32_t ticks) { // Trigger culling of PeerInfo's every hour. This should be called // before the is_open check to ensure that stopped torrents reduce // their memory usage. if (ticks % 120 == 0)// if (ticks % 1 == 0) m_main.peer_list()->cull_peers(PeerList::cull_old || PeerList::cull_keep_interesting); if (!info()->is_open()) return; // Every 2 minutes. if (ticks % 4 == 0) { for (ConnectionList::iterator itr = m_main.connection_list()->begin(); itr != m_main.connection_list()->end(); ) if (!(*itr)->receive_keepalive()) itr = m_main.connection_list()->erase(itr, ConnectionList::disconnect_available); else itr++; } m_main.receive_connect_peers();}voidDownloadWrapper::receive_update_priorities() { if (m_main.chunk_selector()->empty()) return; m_main.chunk_selector()->high_priority()->clear(); m_main.chunk_selector()->normal_priority()->clear(); for (EntryList::iterator itr = m_main.content()->entry_list()->begin(); itr != m_main.content()->entry_list()->end(); ++itr) { if ((*itr)->priority() == 1) m_main.chunk_selector()->normal_priority()->insert((*itr)->range().first, (*itr)->range().second); else if ((*itr)->priority() == 2) m_main.chunk_selector()->high_priority()->insert((*itr)->range().first, (*itr)->range().second); } m_main.chunk_selector()->update_priorities(); std::for_each(m_main.connection_list()->begin(), m_main.connection_list()->end(), std::mem_fun(&PeerConnectionBase::update_interested));}voidDownloadWrapper::finished_download() { // We delay emitting the signal to allow the delegator to // clean up. If we do a straight call it would cause problems // for clients that wish to close and reopen the torrent, as // HashQueue, Delegator etc shouldn't be cleaned up at this // point. // // This needs to be seperated into a new function. if (!m_delayDownloadDone.is_queued()) priority_queue_insert(&taskScheduler, &m_delayDownloadDone, cachedTime); m_main.connection_list()->erase_seeders(); info()->down_rate()->reset_rate();}}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -