2009-01-07 06:53:29 +00:00
|
|
|
/*
|
2021-10-29 18:24:30 +00:00
|
|
|
* This file Copyright (C) 2009-2021 Mnemosyne LLC
|
2007-12-25 05:37:32 +00:00
|
|
|
*
|
2014-01-21 03:10:30 +00:00
|
|
|
* It may be used under the GNU GPL versions 2 or 3
|
2014-01-19 01:09:44 +00:00
|
|
|
* or any future license endorsed by Mnemosyne LLC.
|
2007-12-25 05:37:32 +00:00
|
|
|
*
|
2009-01-07 06:53:29 +00:00
|
|
|
*/
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-11-14 20:21:28 +00:00
|
|
|
#pragma once
|
|
|
|
|
2008-11-24 20:17:36 +00:00
|
|
|
#ifndef __TRANSMISSION__
|
2017-04-19 12:04:45 +00:00
|
|
|
#error only libtransmission should #include this header.
|
2008-11-24 20:17:36 +00:00
|
|
|
#endif
|
|
|
|
|
2021-10-22 02:40:55 +00:00
|
|
|
#include <optional>
|
2021-09-29 19:52:19 +00:00
|
|
|
#include <string>
|
2021-10-20 02:30:50 +00:00
|
|
|
#include <string_view>
|
2021-10-29 18:24:30 +00:00
|
|
|
#include <unordered_map>
|
2021-09-29 19:52:19 +00:00
|
|
|
#include <unordered_set>
|
2021-10-16 14:04:19 +00:00
|
|
|
#include <vector>
|
2021-09-29 19:52:19 +00:00
|
|
|
|
2011-03-31 14:53:22 +00:00
|
|
|
#include "bandwidth.h" /* tr_bandwidth */
|
2021-10-29 18:24:30 +00:00
|
|
|
#include "bitfield.h"
|
2009-01-02 17:01:55 +00:00
|
|
|
#include "completion.h" /* tr_completion */
|
2021-10-31 16:38:10 +00:00
|
|
|
#include "file.h"
|
2021-11-02 23:00:01 +00:00
|
|
|
#include "quark.h"
|
2017-04-21 07:40:57 +00:00
|
|
|
#include "session.h" /* tr_sessionLock(), tr_sessionUnlock() */
|
2017-06-08 07:24:12 +00:00
|
|
|
#include "tr-assert.h"
|
2020-08-11 18:11:55 +00:00
|
|
|
#include "tr-macros.h"
|
2009-08-13 17:25:26 +00:00
|
|
|
#include "utils.h" /* TR_GNUC_PRINTF */
|
2009-01-02 17:01:55 +00:00
|
|
|
|
2021-10-10 00:13:40 +00:00
|
|
|
class tr_swarm;
|
2021-10-29 18:24:30 +00:00
|
|
|
struct tr_torrent;
|
2009-09-25 21:05:59 +00:00
|
|
|
struct tr_torrent_tiers;
|
2009-11-24 02:16:31 +00:00
|
|
|
struct tr_magnet_info;
|
2008-11-06 02:56:51 +00:00
|
|
|
|
2008-10-20 16:00:14 +00:00
|
|
|
/**
|
|
|
|
*** Package-visible ctor API
|
|
|
|
**/
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentFree(tr_torrent* tor);
|
2012-12-14 20:04:37 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_ctorSetSave(tr_ctor* ctor, bool saveMetadataInOurTorrentsDir);
|
2008-10-20 16:00:14 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
bool tr_ctorGetSave(tr_ctor const* ctor);
|
2008-10-20 16:00:14 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
void tr_ctorInitTorrentPriorities(tr_ctor const* ctor, tr_torrent* tor);
|
2009-04-02 20:43:42 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
void tr_ctorInitTorrentWanted(tr_ctor const* ctor, tr_torrent* tor);
|
2009-04-02 20:43:42 +00:00
|
|
|
|
2008-10-20 16:00:14 +00:00
|
|
|
/**
|
|
|
|
***
|
|
|
|
**/
|
|
|
|
|
2007-12-25 05:37:32 +00:00
|
|
|
/* just like tr_torrentSetFileDLs but doesn't trigger a fastresume save */
|
2017-04-20 16:02:19 +00:00
|
|
|
void tr_torrentInitFileDLs(tr_torrent* tor, tr_file_index_t const* files, tr_file_index_t fileCount, bool do_download);
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2021-09-29 19:52:19 +00:00
|
|
|
using tr_labels_t = std::unordered_set<std::string>;
|
|
|
|
|
|
|
|
void tr_torrentSetLabels(tr_torrent* tor, tr_labels_t&& labels);
|
2019-02-17 09:33:57 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentRecheckCompleteness(tr_torrent*);
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentSetHasPiece(tr_torrent* tor, tr_piece_index_t pieceIndex, bool has);
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentChangeMyPort(tr_torrent* session);
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2021-11-02 00:18:09 +00:00
|
|
|
tr_torrent* tr_torrentFindFromHashString(tr_session* session, std::string_view hash_string);
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
tr_torrent* tr_torrentFindFromObfuscatedHash(tr_session* session, uint8_t const* hash);
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
bool tr_torrentIsPieceTransferAllowed(tr_torrent const* torrent, tr_direction direction);
|
2008-11-07 04:10:27 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
#define tr_block(a, b) _tr_block(tor, a, b)
|
2009-02-13 18:23:56 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
tr_block_index_t _tr_block(tr_torrent const* tor, tr_piece_index_t index, uint32_t offset);
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
bool tr_torrentReqIsValid(tr_torrent const* tor, tr_piece_index_t index, uint32_t offset, uint32_t length);
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
uint64_t tr_pieceOffset(tr_torrent const* tor, tr_piece_index_t index, uint32_t offset, uint32_t length);
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2021-08-15 09:41:48 +00:00
|
|
|
void tr_torrentGetBlockLocation(
|
|
|
|
tr_torrent const* tor,
|
|
|
|
tr_block_index_t block,
|
|
|
|
tr_piece_index_t* piece,
|
|
|
|
uint32_t* offset,
|
2017-04-19 12:04:45 +00:00
|
|
|
uint32_t* length);
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2021-10-25 03:39:19 +00:00
|
|
|
struct tr_block_range
|
|
|
|
{
|
|
|
|
tr_block_index_t first;
|
|
|
|
tr_block_index_t last;
|
|
|
|
};
|
2011-02-23 03:54:04 +00:00
|
|
|
|
2021-10-25 03:39:19 +00:00
|
|
|
tr_block_range tr_torGetFileBlockRange(tr_torrent const* tor, tr_file_index_t const file);
|
|
|
|
|
|
|
|
tr_block_range tr_torGetPieceBlockRange(tr_torrent const* tor, tr_piece_index_t const piece);
|
2011-02-23 03:54:04 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentInitFilePriority(tr_torrent* tor, tr_file_index_t fileIndex, tr_priority_t priority);
|
2011-02-23 03:54:04 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentCheckSeedLimit(tr_torrent* tor);
|
2009-03-01 13:56:22 +00:00
|
|
|
|
2009-08-13 14:47:56 +00:00
|
|
|
/** save a torrent's .resume file if it's changed since the last time it was saved */
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentSave(tr_torrent* tor);
|
2009-03-01 13:56:22 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
void tr_torrentSetLocalError(tr_torrent* tor, char const* fmt, ...) TR_GNUC_PRINTF(2, 3);
|
2009-01-13 16:39:19 +00:00
|
|
|
|
fix: gcc warnings in libtransmission/ and utils/ (#843)
* fix: __attribute__(__printf__) warnings
* fix: implicit fallthrough warning
* fixup! fix: implicit fallthrough warning
* fix: disable warnings for 3rd party code
Since we want to leave upstream code as-is
* fixup! fix: disable warnings for 3rd party code
* fixup! fix: disable warnings for 3rd party code
* silence spurious alignment warning
Xrefs
Discussion: https://stackoverflow.com/a/35554349
Macro inspiration: https://pagure.io/SSSD/sssd/blob/90ac46f71068d131391492360a8553bdd005b5a7/f/src/util/util_safealign.h#_35
* fixup! fix: disable warnings for 3rd party code
* fixup! fix: implicit fallthrough warning
* make uncrustify happy
* remove uncrustify-test.sh
that's probably off-topic for this PR
* fixup! fix: __attribute__(__printf__) warnings
* Update libtransmission/CMakeLists.txt
Co-Authored-By: ckerr <ckerr@github.com>
* fixup! silence spurious alignment warning
* use -w for DISABLE_WARNINGS in Clang
* refactor: fix libtransmission deprecation warnings
* fix: pthread_create's start_routine's return value
This was defined as `void` on non-Windows but should have been `void*`
* chore: uncrustify
* fix: add DISABLE_WARNINGS option for SunPro Studio
* fix "unused in lambda capture" warnings by clang++
* fix 'increases required alignment' warning
Caused from storing int16_t's in a char array.
* fix net.c 'increases required alignment' warning
The code passes in a `struct sockaddr_storage*` which is a padded struct
large enough for the necessary alignment. Unfortunately it was recast as
a `struct sockaddr*` which has less padding and a smaller alignment. The
warning occrred because of these differing alignments.
* make building quieter so warnings are more visible
* fixup! fix 'increases required alignment' warning
* Fix -Wcast-function-type warnings in GTK+ app code
https://gitlab.gnome.org/GNOME/gnome-terminal/issues/96 talks about both
the issue and its solution.
GCC 8's -Wcast-function-type, enabled by -Wextra, is problematic in glib
applications because it's idiomatic there to recast function signatures,
e.g. `g_slist_free(list, (GFunc)g_free, NULL);`.
Disabling the warning with pragmas causes "unrecognized pragma" warnings
on clang and older versions of gcc, and disabling the warning could miss
actual bugs. GCC defines `void (*)(void)` as a special case that matches
anything so we can silence warnings by double-casting through GCallback.
In the previous example, the warning is silenced by changing the code to
read `g_slist_free(list, (GFunc)(GCallback)g_free, NULL);`).
* fixup! fix "unused in lambda capture" warnings by clang++
* fixup! fix "unused in lambda capture" warnings by clang++
* fix two more libtransmission compiler warnings
* fix: in watchdir, use TR_ENABLE_ASSERTS not NDEBUG
2019-11-06 17:27:03 +00:00
|
|
|
void tr_torrentSetDateAdded(tr_torrent* torrent, time_t addedDate);
|
|
|
|
|
|
|
|
void tr_torrentSetDateActive(tr_torrent* torrent, time_t activityDate);
|
|
|
|
|
|
|
|
void tr_torrentSetDateDone(tr_torrent* torrent, time_t doneDate);
|
|
|
|
|
2020-10-13 15:33:56 +00:00
|
|
|
/** Return the mime-type (e.g. "audio/x-flac") that matches more of the
|
|
|
|
torrent's content than any other mime-type. */
|
2021-10-20 02:30:50 +00:00
|
|
|
std::string_view tr_torrentPrimaryMimeType(tr_torrent const* tor);
|
2020-10-13 15:33:56 +00:00
|
|
|
|
2021-10-06 14:26:07 +00:00
|
|
|
enum tr_verify_state
|
2007-12-25 05:37:32 +00:00
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
TR_VERIFY_NONE,
|
|
|
|
TR_VERIFY_WAIT,
|
|
|
|
TR_VERIFY_NOW
|
2021-10-06 14:26:07 +00:00
|
|
|
};
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentSetVerifyState(tr_torrent* tor, tr_verify_state state);
|
2009-03-26 18:06:54 +00:00
|
|
|
|
2017-04-21 07:40:57 +00:00
|
|
|
tr_torrent_activity tr_torrentGetActivity(tr_torrent const* tor);
|
2010-05-01 16:04:00 +00:00
|
|
|
|
2009-11-24 02:16:31 +00:00
|
|
|
struct tr_incomplete_metadata;
|
|
|
|
|
2010-01-19 19:37:00 +00:00
|
|
|
/** @brief Torrent object */
|
2007-12-25 05:37:32 +00:00
|
|
|
struct tr_torrent
|
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_session* session;
|
|
|
|
tr_info info;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
int magicNumber;
|
2009-01-30 00:41:08 +00:00
|
|
|
|
2021-10-29 18:24:30 +00:00
|
|
|
std::optional<double> verify_progress;
|
|
|
|
std::vector<tr_sha1_digest_t> piece_checksums;
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_stat_errtype error;
|
|
|
|
char errorString[128];
|
2021-11-02 23:00:01 +00:00
|
|
|
tr_quark error_announce_url;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2021-10-29 18:24:30 +00:00
|
|
|
/// DND
|
|
|
|
|
|
|
|
tr_bitfield dnd_pieces_ = tr_bitfield{ 0 };
|
|
|
|
|
|
|
|
bool pieceIsDnd(tr_piece_index_t piece) const
|
|
|
|
{
|
|
|
|
return dnd_pieces_.test(piece);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// PRIORITIES
|
|
|
|
|
|
|
|
// since 'TR_PRI_NORMAL' is by far the most common, save some
|
|
|
|
// space by treating anything not in the map as normal
|
|
|
|
std::unordered_map<tr_piece_index_t, tr_priority_t> piece_priorities_;
|
|
|
|
|
|
|
|
void setPiecePriority(tr_piece_index_t piece, tr_priority_t priority)
|
|
|
|
{
|
|
|
|
if (priority == TR_PRI_NORMAL)
|
|
|
|
{
|
|
|
|
piece_priorities_.erase(piece);
|
|
|
|
|
|
|
|
if (std::empty(piece_priorities_))
|
|
|
|
{
|
|
|
|
// ensure we release piece_priorities_' internal memory
|
|
|
|
piece_priorities_ = decltype(piece_priorities_){};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
piece_priorities_[piece] = priority;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tr_priority_t piecePriority(tr_piece_index_t piece) const
|
|
|
|
{
|
|
|
|
auto const it = piece_priorities_.find(piece);
|
|
|
|
if (it == std::end(piece_priorities_))
|
|
|
|
{
|
|
|
|
return TR_PRI_NORMAL;
|
|
|
|
}
|
|
|
|
return it->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// CHECKSUMS
|
|
|
|
|
|
|
|
tr_bitfield checked_pieces_ = tr_bitfield{ 0 };
|
|
|
|
|
|
|
|
bool checkPiece(tr_piece_index_t piece);
|
|
|
|
|
|
|
|
bool ensurePieceIsChecked(tr_piece_index_t piece)
|
|
|
|
{
|
|
|
|
TR_ASSERT(piece < info.pieceCount);
|
|
|
|
|
|
|
|
if (checked_pieces_.test(piece))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool const checked = checkPiece(piece);
|
|
|
|
this->anyDate = tr_time();
|
|
|
|
this->setDirty();
|
|
|
|
|
|
|
|
checked_pieces_.set(piece, checked);
|
|
|
|
return checked;
|
|
|
|
}
|
|
|
|
|
|
|
|
void initCheckedPieces(tr_bitfield const& checked, time_t const* mtimes /*fileCount*/)
|
|
|
|
{
|
|
|
|
TR_ASSERT(std::size(checked) == info.pieceCount);
|
|
|
|
checked_pieces_ = checked;
|
|
|
|
|
2021-10-31 16:38:10 +00:00
|
|
|
auto filename = std::string{};
|
2021-10-29 18:24:30 +00:00
|
|
|
for (size_t i = 0; i < info.fileCount; ++i)
|
|
|
|
{
|
2021-10-31 16:38:10 +00:00
|
|
|
auto const found = this->findFile(filename, i);
|
|
|
|
auto const mtime = found ? found->last_modified_at : 0;
|
2021-10-29 18:24:30 +00:00
|
|
|
|
|
|
|
info.files[i].mtime = mtime;
|
|
|
|
|
|
|
|
// if a file has changed, mark its pieces as unchecked
|
2021-10-31 16:38:10 +00:00
|
|
|
if (mtime == 0 || mtime != mtimes[i])
|
2021-10-29 18:24:30 +00:00
|
|
|
{
|
|
|
|
checked_pieces_.unsetRange(info.files[i].firstPiece, info.files[i].lastPiece);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-31 16:38:10 +00:00
|
|
|
/// FINDING FILES
|
|
|
|
|
|
|
|
struct tr_found_file_t : public tr_sys_path_info
|
|
|
|
{
|
|
|
|
std::string& filename; // /home/foo/Downloads/torrent/01-file-one.txt
|
|
|
|
std::string_view base; // /home/foo/Downloads
|
|
|
|
std::string_view subpath; // /torrent/01-file-one.txt
|
|
|
|
|
|
|
|
tr_found_file_t(tr_sys_path_info info, std::string& f, std::string_view b)
|
|
|
|
: tr_sys_path_info{ info }
|
|
|
|
, filename{ f }
|
|
|
|
, base{ b }
|
|
|
|
, subpath{ f.c_str() + std::size(b) + 1 }
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
std::optional<tr_found_file_t> findFile(std::string& filename, tr_file_index_t i) const;
|
|
|
|
|
2021-10-29 18:24:30 +00:00
|
|
|
///
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
uint8_t obfuscatedHash[SHA_DIGEST_LENGTH];
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2009-11-24 02:16:31 +00:00
|
|
|
/* Used when the torrent has been created with a magnet link
|
|
|
|
* and we're in the process of downloading the metainfo from
|
|
|
|
* other peers */
|
2017-04-19 12:04:45 +00:00
|
|
|
struct tr_incomplete_metadata* incompleteMetadata;
|
2009-11-24 02:16:31 +00:00
|
|
|
|
2008-12-04 05:27:59 +00:00
|
|
|
/* If the initiator of the connection receives a handshake in which the
|
|
|
|
* peer_id does not match the expected peerid, then the initiator is
|
|
|
|
* expected to drop the connection. Note that the initiator presumably
|
|
|
|
* received the peer information from the tracker, which includes the
|
|
|
|
* peer_id that was registered by the peer. The peer_id from the tracker
|
|
|
|
* and in the handshake are expected to match.
|
|
|
|
*/
|
2021-10-22 02:40:55 +00:00
|
|
|
std::optional<tr_peer_id_t> peer_id;
|
2013-02-02 13:42:50 +00:00
|
|
|
|
|
|
|
time_t peer_id_creation_time;
|
2008-12-04 05:27:59 +00:00
|
|
|
|
2009-10-19 05:05:00 +00:00
|
|
|
/* Where the files will be when it's complete */
|
2017-04-19 12:04:45 +00:00
|
|
|
char* downloadDir;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2009-10-19 05:05:00 +00:00
|
|
|
/* Where the files are when the torrent is incomplete */
|
2017-04-19 12:04:45 +00:00
|
|
|
char* incompleteDir;
|
2009-10-19 05:05:00 +00:00
|
|
|
|
2010-06-16 03:05:23 +00:00
|
|
|
/* Length, in bytes, of the "info" dict in the .torrent file. */
|
2015-12-25 11:34:35 +00:00
|
|
|
size_t infoDictLength;
|
2009-11-24 02:16:31 +00:00
|
|
|
|
2010-06-16 03:05:23 +00:00
|
|
|
/* Offset, in bytes, of the beginning of the "info" dict in the .torrent file.
|
|
|
|
*
|
|
|
|
* Used by the torrent-magnet code for serving metainfo to peers.
|
|
|
|
* This field is lazy-generated and might not be initialized yet. */
|
2015-12-25 11:34:35 +00:00
|
|
|
size_t infoDictOffset;
|
2009-11-24 02:16:31 +00:00
|
|
|
|
2009-10-19 05:05:00 +00:00
|
|
|
/* Where the files are now.
|
|
|
|
* This pointer will be equal to downloadDir or incompleteDir */
|
2017-04-20 16:02:19 +00:00
|
|
|
char const* currentDir;
|
2009-10-19 05:05:00 +00:00
|
|
|
|
2007-12-25 05:37:32 +00:00
|
|
|
/* How many bytes we ask for per request */
|
2017-04-19 12:04:45 +00:00
|
|
|
uint32_t blockSize;
|
|
|
|
tr_block_index_t blockCount;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
uint32_t lastBlockSize;
|
|
|
|
uint32_t lastPieceSize;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2021-10-19 00:25:00 +00:00
|
|
|
uint32_t blockCountInPiece;
|
|
|
|
uint32_t blockCountInLastPiece;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
struct tr_completion completion;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_completeness completeness;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
struct tr_torrent_tiers* tiers;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
time_t dhtAnnounceAt;
|
|
|
|
time_t dhtAnnounce6At;
|
|
|
|
bool dhtAnnounceInProgress;
|
|
|
|
bool dhtAnnounce6InProgress;
|
2010-12-12 16:43:19 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
time_t lpdAnnounceAt;
|
2009-05-19 18:38:26 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
uint64_t downloadedCur;
|
|
|
|
uint64_t downloadedPrev;
|
|
|
|
uint64_t uploadedCur;
|
|
|
|
uint64_t uploadedPrev;
|
|
|
|
uint64_t corruptCur;
|
|
|
|
uint64_t corruptPrev;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
uint64_t etaDLSpeedCalculatedAt;
|
|
|
|
unsigned int etaDLSpeed_Bps;
|
|
|
|
uint64_t etaULSpeedCalculatedAt;
|
|
|
|
unsigned int etaULSpeed_Bps;
|
2009-07-02 01:50:07 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
time_t activityDate;
|
2019-11-12 23:13:42 +00:00
|
|
|
time_t addedDate;
|
|
|
|
time_t anyDate;
|
2017-04-19 12:04:45 +00:00
|
|
|
time_t doneDate;
|
2019-11-12 23:13:42 +00:00
|
|
|
time_t editDate;
|
2017-04-19 12:04:45 +00:00
|
|
|
time_t startDate;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
int secondsDownloading;
|
|
|
|
int secondsSeeding;
|
2010-12-23 19:32:59 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
int queuePosition;
|
2011-08-01 22:24:24 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_torrent_metadata_func metadata_func;
|
|
|
|
void* metadata_func_user_data;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_torrent_completeness_func completeness_func;
|
|
|
|
void* completeness_func_user_data;
|
2009-11-26 17:58:44 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_torrent_ratio_limit_hit_func ratio_limit_hit_func;
|
|
|
|
void* ratio_limit_hit_func_user_data;
|
2009-02-13 18:23:56 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_torrent_idle_limit_hit_func idle_limit_hit_func;
|
|
|
|
void* idle_limit_hit_func_user_data;
|
2010-07-16 03:12:57 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void* queue_started_user_data;
|
2021-08-15 09:41:48 +00:00
|
|
|
void (*queue_started_callback)(tr_torrent*, void* queue_started_user_data);
|
2011-08-03 03:14:57 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
bool isRunning;
|
|
|
|
bool isStopping;
|
|
|
|
bool isDeleting;
|
|
|
|
bool startAfterVerify;
|
|
|
|
bool isDirty;
|
2021-10-29 18:24:30 +00:00
|
|
|
|
|
|
|
void setDirty()
|
|
|
|
{
|
|
|
|
this->isDirty = true;
|
|
|
|
}
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
bool isQueued;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2019-12-15 16:18:11 +00:00
|
|
|
bool prefetchMagnetMetadata;
|
2017-04-19 12:04:45 +00:00
|
|
|
bool magnetVerify;
|
2014-01-19 05:03:27 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
bool infoDictOffsetIsCached;
|
2010-06-16 03:05:23 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
uint16_t maxConnectedPeers;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_verify_state verifyState;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
time_t lastStatTime;
|
|
|
|
tr_stat stats;
|
2007-12-25 05:37:32 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
int uniqueId;
|
2008-09-17 19:44:24 +00:00
|
|
|
|
2021-10-09 12:52:09 +00:00
|
|
|
// Changed to non-owning pointer temporarily till tr_torrent becomes C++-constructible and destructible
|
|
|
|
// TODO: change tr_bandwidth* to owning pointer to the bandwidth, or remove * and own the value
|
2021-10-10 01:12:03 +00:00
|
|
|
Bandwidth* bandwidth;
|
2009-06-16 19:31:17 +00:00
|
|
|
|
2021-10-10 00:13:40 +00:00
|
|
|
tr_swarm* swarm;
|
2009-02-13 18:23:56 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
float desiredRatio;
|
|
|
|
tr_ratiolimit ratioLimitMode;
|
2009-08-13 17:50:41 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
uint16_t idleLimitMinutes;
|
|
|
|
tr_idlelimit idleLimitMode;
|
|
|
|
bool finishedSeedingByIdle;
|
2019-02-17 09:33:57 +00:00
|
|
|
|
2021-09-29 19:52:19 +00:00
|
|
|
tr_labels_t labels;
|
2007-12-25 05:37:32 +00:00
|
|
|
};
|
|
|
|
|
2009-01-02 06:28:22 +00:00
|
|
|
/* what piece index is this block in? */
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr tr_piece_index_t tr_torBlockPiece(tr_torrent const* tor, tr_block_index_t const block)
|
2009-01-02 06:28:22 +00:00
|
|
|
{
|
|
|
|
return block / tor->blockCountInPiece;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* how many bytes are in this piece? */
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr uint32_t tr_torPieceCountBytes(tr_torrent const* tor, tr_piece_index_t const piece)
|
2009-01-02 06:28:22 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return piece + 1 == tor->info.pieceCount ? tor->lastPieceSize : tor->info.pieceSize;
|
2009-01-02 06:28:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* how many bytes are in this block? */
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr uint32_t tr_torBlockCountBytes(tr_torrent const* tor, tr_block_index_t const block)
|
2009-01-02 06:28:22 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return block + 1 == tor->blockCount ? tor->lastBlockSize : tor->blockSize;
|
2009-01-02 06:28:22 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline void tr_torrentLock(tr_torrent const* tor)
|
2015-01-02 11:15:31 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_sessionLock(tor->session);
|
2015-01-02 11:15:31 +00:00
|
|
|
}
|
2017-04-19 12:04:45 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline bool tr_torrentIsLocked(tr_torrent const* tor)
|
2011-01-18 23:44:36 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_sessionIsLocked(tor->session);
|
2011-01-18 23:44:36 +00:00
|
|
|
}
|
2017-04-19 12:04:45 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline void tr_torrentUnlock(tr_torrent const* tor)
|
2009-01-02 17:01:55 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
tr_sessionUnlock(tor->session);
|
2009-01-02 17:01:55 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline bool tr_torrentExists(tr_session const* session, uint8_t const* torrentHash)
|
2009-01-02 17:01:55 +00:00
|
|
|
{
|
2021-09-15 00:18:09 +00:00
|
|
|
return tr_torrentFindFromHash((tr_session*)session, torrentHash) != nullptr;
|
2009-01-02 17:01:55 +00:00
|
|
|
}
|
|
|
|
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr tr_completeness tr_torrentGetCompleteness(tr_torrent const* tor)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
|
|
|
return tor->completeness;
|
|
|
|
}
|
|
|
|
|
2021-11-03 02:55:33 +00:00
|
|
|
// TODO: rename this to tr_torrentIsDone()? both seed and partial seed return true
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr bool tr_torrentIsSeed(tr_torrent const* tor)
|
2009-01-03 00:25:27 +00:00
|
|
|
{
|
2013-08-18 13:06:39 +00:00
|
|
|
return tr_torrentGetCompleteness(tor) != TR_LEECH;
|
2009-01-03 00:25:27 +00:00
|
|
|
}
|
|
|
|
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr bool tr_torrentIsPrivate(tr_torrent const* tor)
|
2009-01-03 00:25:27 +00:00
|
|
|
{
|
2021-09-15 00:18:09 +00:00
|
|
|
return tor != nullptr && tor->info.isPrivate;
|
2009-01-03 00:25:27 +00:00
|
|
|
}
|
|
|
|
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr bool tr_torrentAllowsPex(tr_torrent const* tor)
|
2009-01-03 00:25:27 +00:00
|
|
|
{
|
2021-09-15 00:18:09 +00:00
|
|
|
return tor != nullptr && tor->session->isPexEnabled && !tr_torrentIsPrivate(tor);
|
2009-01-03 00:25:27 +00:00
|
|
|
}
|
|
|
|
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr bool tr_torrentAllowsDHT(tr_torrent const* tor)
|
2009-05-19 18:38:26 +00:00
|
|
|
{
|
2021-09-15 00:18:09 +00:00
|
|
|
return tor != nullptr && tr_sessionAllowsDHT(tor->session) && !tr_torrentIsPrivate(tor);
|
2009-05-19 18:38:26 +00:00
|
|
|
}
|
|
|
|
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr bool tr_torrentAllowsLPD(tr_torrent const* tor)
|
2010-05-01 16:04:00 +00:00
|
|
|
{
|
2021-09-15 00:18:09 +00:00
|
|
|
return tor != nullptr && tr_sessionAllowsLPD(tor->session) && !tr_torrentIsPrivate(tor);
|
2010-05-01 16:04:00 +00:00
|
|
|
}
|
|
|
|
|
2009-01-30 00:41:08 +00:00
|
|
|
/***
|
|
|
|
****
|
|
|
|
***/
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
TORRENT_MAGIC_NUMBER = 95549
|
|
|
|
};
|
|
|
|
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr bool tr_isTorrent(tr_torrent const* tor)
|
2009-01-30 00:41:08 +00:00
|
|
|
{
|
2021-09-15 00:18:09 +00:00
|
|
|
return tor != nullptr && tor->magicNumber == TORRENT_MAGIC_NUMBER && tr_isSession(tor->session);
|
2009-01-30 00:41:08 +00:00
|
|
|
}
|
|
|
|
|
2009-08-07 05:29:37 +00:00
|
|
|
/* set a flag indicating that the torrent's .resume file
|
|
|
|
* needs to be saved when the torrent is closed */
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr void tr_torrentSetDirty(tr_torrent* tor)
|
2009-08-07 05:29:37 +00:00
|
|
|
{
|
2017-06-08 07:24:12 +00:00
|
|
|
TR_ASSERT(tr_isTorrent(tor));
|
2009-08-07 05:29:37 +00:00
|
|
|
|
2011-03-22 15:19:54 +00:00
|
|
|
tor->isDirty = true;
|
2009-08-07 05:29:37 +00:00
|
|
|
}
|
|
|
|
|
2019-11-12 23:13:42 +00:00
|
|
|
/* note that the torrent's tr_info just changed */
|
|
|
|
static inline void tr_torrentMarkEdited(tr_torrent* tor)
|
|
|
|
{
|
|
|
|
TR_ASSERT(tr_isTorrent(tor));
|
|
|
|
|
|
|
|
tor->editDate = tr_time();
|
|
|
|
}
|
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
uint32_t tr_getBlockSize(uint32_t pieceSize);
|
2010-06-24 20:36:05 +00:00
|
|
|
|
2009-10-19 05:05:00 +00:00
|
|
|
/**
|
2013-01-26 23:08:51 +00:00
|
|
|
* Tell the tr_torrent that it's gotten a block
|
2009-10-19 05:05:00 +00:00
|
|
|
*/
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentGotBlock(tr_torrent* tor, tr_block_index_t blockIndex);
|
2009-10-19 05:05:00 +00:00
|
|
|
|
2009-10-20 04:43:51 +00:00
|
|
|
/**
|
2020-11-02 15:16:12 +00:00
|
|
|
* @brief Like tr_torrentFindFile(), but splits the filename into base and subpath.
|
2009-10-20 04:43:51 +00:00
|
|
|
*
|
2021-10-06 16:32:17 +00:00
|
|
|
* If the file is found, "tr_buildPath(base, subpath, nullptr)"
|
2009-10-20 04:43:51 +00:00
|
|
|
* will generate the complete filename.
|
|
|
|
*
|
|
|
|
* @return true if the file is found, false otherwise.
|
|
|
|
*
|
|
|
|
* @param base if the torrent is found, this will be either
|
|
|
|
* tor->downloadDir or tor->incompleteDir
|
|
|
|
* @param subpath on success, this pointer is assigned a newly-allocated
|
|
|
|
* string holding the second half of the filename.
|
|
|
|
*/
|
2017-04-20 16:02:19 +00:00
|
|
|
bool tr_torrentFindFile2(tr_torrent const*, tr_file_index_t fileNo, char const** base, char** subpath, time_t* mtime);
|
2009-10-20 03:14:44 +00:00
|
|
|
|
2009-10-20 04:43:51 +00:00
|
|
|
/* Returns a newly-allocated version of the tr_file.name string
|
|
|
|
* that's been modified to denote that it's not a complete file yet.
|
|
|
|
* In the current implementation this is done by appending ".part"
|
|
|
|
* a la Firefox. */
|
2017-04-20 16:02:19 +00:00
|
|
|
char* tr_torrentBuildPartial(tr_torrent const*, tr_file_index_t fileNo);
|
2009-10-20 03:14:44 +00:00
|
|
|
|
2009-11-24 02:16:31 +00:00
|
|
|
/* for when the info dict has been fundamentally changed wrt files,
|
|
|
|
* piece size, etc. such as in BEP 9 where peers exchange metadata */
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentGotNewInfoDict(tr_torrent* tor);
|
2009-10-20 03:14:44 +00:00
|
|
|
|
2017-04-19 12:04:45 +00:00
|
|
|
void tr_torrentSetSpeedLimit_Bps(tr_torrent*, tr_direction, unsigned int Bps);
|
2017-04-20 16:02:19 +00:00
|
|
|
unsigned int tr_torrentGetSpeedLimit_Bps(tr_torrent const*, tr_direction);
|
2010-07-04 06:07:21 +00:00
|
|
|
|
2010-12-09 20:43:23 +00:00
|
|
|
/**
|
|
|
|
* @brief Test a piece against its info dict checksum
|
|
|
|
* @return true if the piece's passes the checksum test
|
2010-12-12 16:43:19 +00:00
|
|
|
*/
|
2017-04-19 12:04:45 +00:00
|
|
|
bool tr_torrentCheckPiece(tr_torrent* tor, tr_piece_index_t pieceIndex);
|
2010-12-09 20:43:23 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
uint64_t tr_torrentGetCurrentSizeOnDisk(tr_torrent const* tor);
|
2010-12-09 20:43:23 +00:00
|
|
|
|
2021-10-22 02:40:55 +00:00
|
|
|
tr_peer_id_t const& tr_torrentGetPeerId(tr_torrent* tor);
|
2013-02-02 13:42:50 +00:00
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline uint64_t tr_torrentGetLeftUntilDone(tr_torrent const* tor)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_cpLeftUntilDone(&tor->completion);
|
2013-08-18 13:06:39 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline bool tr_torrentHasAll(tr_torrent const* tor)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_cpHasAll(&tor->completion);
|
2013-08-18 13:06:39 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline bool tr_torrentHasNone(tr_torrent const* tor)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_cpHasNone(&tor->completion);
|
2013-08-18 13:06:39 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline bool tr_torrentPieceIsComplete(tr_torrent const* tor, tr_piece_index_t i)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_cpPieceIsComplete(&tor->completion, i);
|
2013-08-18 13:06:39 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline bool tr_torrentBlockIsComplete(tr_torrent const* tor, tr_block_index_t i)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_cpBlockIsComplete(&tor->completion, i);
|
2013-08-18 13:06:39 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline size_t tr_torrentMissingBlocksInPiece(tr_torrent const* tor, tr_piece_index_t i)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_cpMissingBlocksInPiece(&tor->completion, i);
|
2013-08-18 13:06:39 +00:00
|
|
|
}
|
|
|
|
|
2017-04-20 16:02:19 +00:00
|
|
|
static inline size_t tr_torrentMissingBytesInPiece(tr_torrent const* tor, tr_piece_index_t i)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_cpMissingBytesInPiece(&tor->completion, i);
|
2013-08-18 13:06:39 +00:00
|
|
|
}
|
|
|
|
|
2021-10-16 14:04:19 +00:00
|
|
|
static inline std::vector<uint8_t> tr_torrentCreatePieceBitfield(tr_torrent const* tor)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
2021-10-16 14:04:19 +00:00
|
|
|
return tr_cpCreatePieceBitfield(&tor->completion);
|
2013-08-18 13:06:39 +00:00
|
|
|
}
|
|
|
|
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr uint64_t tr_torrentHaveTotal(tr_torrent const* tor)
|
2013-08-18 13:06:39 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_cpHaveTotal(&tor->completion);
|
2013-08-18 13:06:39 +00:00
|
|
|
}
|
|
|
|
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr bool tr_torrentIsQueued(tr_torrent const* tor)
|
2011-08-01 22:24:24 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tor->isQueued;
|
2011-08-01 22:24:24 +00:00
|
|
|
}
|
|
|
|
|
2021-10-06 22:24:04 +00:00
|
|
|
constexpr tr_direction tr_torrentGetQueueDirection(tr_torrent const* tor)
|
2011-08-01 22:24:24 +00:00
|
|
|
{
|
2017-04-19 12:04:45 +00:00
|
|
|
return tr_torrentIsSeed(tor) ? TR_UP : TR_DOWN;
|
2011-08-01 22:24:24 +00:00
|
|
|
}
|