*** empty log message ***

This commit is contained in:
Arvid Norberg 2004-02-26 12:59:01 +00:00
parent d2566ed20a
commit 1e574575b4
7 changed files with 679 additions and 538 deletions

View File

@ -135,13 +135,13 @@ boost.filesystem, boost.date_time and various other boost libraries as well as z
<li>Windows 2000 vc7.1</li>
<li>Linux x86 (debian) GCC 3.0.4, GCC 3.2.3</li>
<li>Windows 2000, msvc6 sp5 (does not support 64-bit values due to problems with operator&lt;&lt;(ostream&amp;, __int64))</li>
<li>Cygwin GCC 3.3.1</li>
</ul>
</blockquote>
<p>Fails on:</p>
<blockquote>
<ul class="simple">
<li>GCC 2.95.4 (<tt class="literal"><span class="pre">std::ios_base</span></tt> is missing)</li>
<li>Cygwin GCC 3.3.1 (builds but crashes)</li>
</ul>
</blockquote>
<p>libtorrent is released under the <a class="reference" href="http://www.opensource.org/licenses/bsd-license.php">BSD-license</a>.</p>
@ -166,10 +166,7 @@ bjam &lt;toolset&gt;
&quot;force conformance in for loop scope&quot; and &quot;treat wchar_t as built-in type&quot; to Yes.</p>
<p>If you're building in developer studio 6, you will probably have to use the previous
version of boost, <a class="reference" href="http://sourceforge.net/project/showfiles.php?group_id=7586&amp;package_id=8041&amp;release_id=178835">boost 1.30.2</a>.</p>
<p>There are two versions of the socket code, one that works with unix systems (and bsd-sockets) and
one that uses winsock. If you're building in windows, the file <tt class="literal"><span class="pre">socket_win.cpp</span></tt> is supposed to
be included in the build while <tt class="literal"><span class="pre">socket_bsd.cpp</span></tt> is supposed to be excluded.</p>
<p>The file abstraction has the same kind of separation. There's one <tt class="literal"><span class="pre">file_win.cpp</span></tt> which
<p>There are two versions of the file abstraction. There's one <tt class="literal"><span class="pre">file_win.cpp</span></tt> which
relies on windows file API that supports files larger than 2 Gigabytes. This does not work
in vc6 for some reason, possibly because it may require windows NT and above. The other file,
<tt class="literal"><span class="pre">file.cpp</span></tt> is the default implementation that simply relies on the standard library's fstream,
@ -249,8 +246,12 @@ the <tt class="literal"><span class="pre">session</span></tt>, it contains the m
<pre class="literal-block">
class session: public boost::noncopyable
{
session(std::pair&lt;int, int&gt; listen_port_range, const fingerprint&amp; print);
session(std::pair&lt;int, int&gt; listen_port_range);
session(
std::pair&lt;int, int&gt; listen_port_range
, const fingerprint&amp; print
, const char* listen_interface = 0);
torrent_handle add_torrent(
const torrent_info&amp; t
@ -262,6 +263,13 @@ class session: public boost::noncopyable
void set_http_settings(const http_settings&amp; settings);
void set_upload_rate_limit(int bytes_per_second);
bool is_listening() const;
unsigned short listen_port() const;
bool listen_on(
std::pair&lt;int, int&gt; const&amp; port_range
, const char* interface = 0);
std::auto_ptr&lt;alert&gt; pop_alert();
void set_severity_level(alert::severity_t s);
@ -290,6 +298,21 @@ fingerprint class.</p>
<p><tt class="literal"><span class="pre">set_upload_rate_limit()</span></tt> set the maximum number of bytes allowed to be
sent to peers per second. This bandwidth is distributed among all the peers. If
you don't want to limit upload rate, you can set this to -1 (the default).</p>
<p><tt class="literal"><span class="pre">is_listening()</span></tt> will tell you wether or not the session has successfully
opened a listening port. If it hasn't, this function will return false, and
then you can use <tt class="literal"><span class="pre">listen_on()</span></tt> to make another try.</p>
<p><tt class="literal"><span class="pre">listen_port()</span></tt> returns the port we ended up listening on. Since you just pass
a port-range to the constructor and to <tt class="literal"><span class="pre">listen_on()</span></tt>, to know which port it
ended up using, you have to ask the session using this function.</p>
<p><tt class="literal"><span class="pre">listen_on()</span></tt> will change the listen port and/or the listen interface. If the
session is already listening on a port, this socket will be closed and a new socket
will be opened with these new settings. The port range is the ports it will try
to listen on, if the first port fails, it will continue trying the next port within
the range and so on. The interface parameter can be left as 0, in that case the
os will decide which interface to listen on, otherwise it should be the ip-address
of the interface you want the listener socket bound to. <tt class="literal"><span class="pre">listen_on()</span></tt> returns true
if it managed to open the socket, and false if it failed. If it fails, it will also
generate an appropriate alert (<a class="reference" href="#listen-failed-alert">listen_failed_alert</a>).</p>
<p>The destructor of session will notify all trackers that our torrents has been shut down.
If some trackers are down, they will timout. All this before the destructor of session
returns. So, it's adviced that any kind of interface (such as windows) are closed before
@ -302,7 +325,11 @@ increase the port number by one and try again. If it still fails it will continu
increasing the port number until it succeeds or has reached the end of the range. If it
fails with all ports, a <a class="reference" href="#listen-failed-alert">listen_failed_alert</a> will be posted and the session thread will
exit. The only thing you can do with your session if this alert is posted is to destruct
it and possibly try again or change the port range.</p>
it and possibly try again or change the port range. The listen interaface string is
the name (ip address) of the interface you want to listen on. If this is left as
0, the os will decide which interface to listen on (works in most cases). All torrents
will use this interface to open outgoing connections on by default. You can change
which interface to use for outgoing connections on a per torrent basis. See <a class="reference" href="#torrent-handle">torrent_handle</a>.</p>
<p>For information about the <tt class="literal"><span class="pre">pop_alert()</span></tt> function, see <a class="reference" href="#alerts">alerts</a>.</p>
</div>
<div class="section" id="parsing-torrent-files">
@ -535,6 +562,8 @@ struct torrent_handle
void set_tracker_login(std::string const&amp; username, std::string const&amp; password);
void use_interface(const char* net_interface);
boost::filsystem::path save_path() const;
void set_max_uploads(int max_uploads);
@ -569,6 +598,8 @@ attempt to upload in return for each download. e.g. if set to 2, the client will
as a standard client.</p>
<p><tt class="literal"><span class="pre">set_tracker_login()</span></tt> sets a username and password that will be sent along in the HTTP-request
of the tracker announce. Set this if the tracker requires authorization.</p>
<p><tt class="literal"><span class="pre">use_interface()</span></tt> sets the network interface this torrent will use when it opens outgoing
connections. By default, it uses the same interface as the <a class="reference" href="#session">session</a> uses to listen on.</p>
<p><tt class="literal"><span class="pre">info_hash()</span></tt> returns the info hash for the torrent.</p>
<p><tt class="literal"><span class="pre">set_max_uploads()</span></tt> sets the maximum number of peers that's unchoked at the same time on this
torrent. If you set this to -1, there will be no limit.</p>
@ -1089,8 +1120,8 @@ user in different ways.</p>
<div class="section" id="listen-failed-alert">
<h2><a name="listen-failed-alert">listen_failed_alert</a></h2>
<p>This alert is generated when none of the ports, given in the port range, to
<a class="reference" href="#session">session</a> can be opened for listening. Without a listening port the session
object will exit its thread. This alert is generated as severity level <tt class="literal"><span class="pre">fatal</span></tt>.</p>
<a class="reference" href="#session">session</a> can be opened for listening. This alert is generated as severity
level <tt class="literal"><span class="pre">fatal</span></tt>.</p>
<pre class="literal-block">
struct listen_failed_alert: alert
{

View File

@ -202,6 +202,13 @@ The ``session`` class has the following synopsis::
void set_http_settings(const http_settings& settings);
void set_upload_rate_limit(int bytes_per_second);
bool is_listening() const;
unsigned short listen_port() const;
bool listen_on(
std::pair<int, int> const& port_range
, const char* interface = 0);
std::auto_ptr<alert> pop_alert();
void set_severity_level(alert::severity_t s);
@ -236,6 +243,24 @@ fingerprint class.
sent to peers per second. This bandwidth is distributed among all the peers. If
you don't want to limit upload rate, you can set this to -1 (the default).
``is_listening()`` will tell you wether or not the session has successfully
opened a listening port. If it hasn't, this function will return false, and
then you can use ``listen_on()`` to make another try.
``listen_port()`` returns the port we ended up listening on. Since you just pass
a port-range to the constructor and to ``listen_on()``, to know which port it
ended up using, you have to ask the session using this function.
``listen_on()`` will change the listen port and/or the listen interface. If the
session is already listening on a port, this socket will be closed and a new socket
will be opened with these new settings. The port range is the ports it will try
to listen on, if the first port fails, it will continue trying the next port within
the range and so on. The interface parameter can be left as 0, in that case the
os will decide which interface to listen on, otherwise it should be the ip-address
of the interface you want the listener socket bound to. ``listen_on()`` returns true
if it managed to open the socket, and false if it failed. If it fails, it will also
generate an appropriate alert (listen_failed_alert_).
The destructor of session will notify all trackers that our torrents has been shut down.
If some trackers are down, they will timout. All this before the destructor of session
returns. So, it's adviced that any kind of interface (such as windows) are closed before
@ -1136,8 +1161,8 @@ listen_failed_alert
-------------------
This alert is generated when none of the ports, given in the port range, to
session_ can be opened for listening. Without a listening port the session
object will exit its thread. This alert is generated as severity level ``fatal``.
session_ can be opened for listening. This alert is generated as severity
level ``fatal``.
::

View File

@ -139,7 +139,7 @@ namespace libtorrent
// the torrent is added to the session
session_impl& m_ses;
boost::mutex m_mutex;
mutable boost::mutex m_mutex;
boost::condition m_cond;
// a list of all torrents that are currently checking
@ -164,9 +164,11 @@ namespace libtorrent
void operator()();
void open_listen_port();
// must be locked to access the data
// in this struct
boost::mutex m_mutex;
mutable boost::mutex m_mutex;
torrent* find_torrent(const sha1_hash& info_hash);
const peer_id& get_peer_id() const { return m_peer_id; }
@ -202,6 +204,8 @@ namespace libtorrent
// them
selector m_selector;
boost::shared_ptr<socket> m_listen_socket;
// the settings for the client
http_settings m_settings;
@ -257,8 +261,25 @@ namespace libtorrent
, const boost::filesystem::path& save_path
, const entry& resume_data = entry());
// TODO: add the ability to change listen-port on the fly
// TODO: make it possible to ask for the current listen port
bool is_listening() const;
// if the listen port failed in some way
// you can retry to listen on another port-
// range with this function. If the listener
// succeeded and is currently listening,
// a call to this function will shut down the
// listen port and reopen it using these new
// properties (the given interface and port range).
// As usual, if the interface is left as 0
// this function will return false on failure.
// If it fails, it will also generate alerts describing
// the error. It will return true on success.
bool listen_on(
std::pair<int, int> const& port_range
, const char* interface = 0);
// returns the port we ended up listening on
unsigned short listen_port() const;
void remove_torrent(const torrent_handle& h);

View File

@ -52,9 +52,6 @@ POSSIBILITY OF SUCH DAMAGE.
#include <string>
// TODO: support ToS
// TODO: support binding to selected interface
// Distinguish between binding listen sockets and
// outgoing connections.
namespace libtorrent
{
@ -102,8 +99,8 @@ namespace libtorrent
unsigned short port;
const static unsigned short any_port = 0;
const static unsigned int any_addr = 0;
BOOST_STATIC_CONSTANT(unsigned short, any_port = 0);
BOOST_STATIC_CONSTANT(unsigned int, any_addr = 0);
private:

View File

@ -169,7 +169,7 @@ namespace libtorrent
// (make a rerequest from the tracker)
void force_reannounce() const;
// TODO: add a feature where the user can ask the torrent
// TODO: add a feature where the user can tell the torrent
// to finish all pieces currently in the pipeline, and then
// abort the torrent.

View File

@ -36,6 +36,10 @@ POSSIBILITY OF SUCH DAMAGE.
#include <algorithm>
#include <boost/limits.hpp>
#if defined(_MSC_VER) && _MSC_VER < 1300
#define for if (false) {} else for
#endif
namespace libtorrent
{
namespace

View File

@ -160,10 +160,8 @@ namespace
*/
}
namespace libtorrent
namespace libtorrent { namespace detail
{
namespace detail
{
void checker_impl::operator()()
{
eh_initializer();
@ -218,6 +216,7 @@ namespace libtorrent
catch(const std::exception& e)
{
// TODO: generate an alert here!
// This will happen if the storage fails to initialize
#ifndef NDEBUG
std::cerr << "error while checking files: " << e.what() << "\n";
#endif
@ -295,6 +294,69 @@ namespace libtorrent
}
}
void session_impl::open_listen_port()
{
try
{
// create listener socket
m_listen_socket = boost::shared_ptr<socket>(new socket(socket::tcp, false));
for(;;)
{
try
{
m_listen_socket->listen(m_listen_interface, 10);
break;
}
catch (network_error& e)
{
if (e.error_code() == socket::address_not_available)
{
std::string msg = "cannot listen on the given interface '" + m_listen_interface.as_string() + "'";
m_alerts.post_alert(listen_failed_alert(msg));
#ifndef NDEBUG
(*m_logger) << msg << "\n";
#endif
m_listen_socket.reset();
break;
}
m_listen_interface.port++;
if (m_listen_interface.port > m_listen_port_range.second)
{
std::stringstream msg;
msg << "none of the ports in the range ["
<< m_listen_port_range.first
<< ", " << m_listen_port_range.second
<< "] could be opened for listening";
m_alerts.post_alert(listen_failed_alert(msg.str()));
#ifndef NDEBUG
(*m_logger) << msg.str() << "\n";
#endif
m_listen_socket.reset();
break;
}
}
}
}
catch (network_error& e)
{
m_alerts.post_alert(listen_failed_alert(e.what()));
}
#ifndef NDEBUG
if (m_listen_socket)
{
(*m_logger) << "listening on port: " << m_listen_interface.port << "\n";
}
#endif
if (m_listen_socket)
{
m_selector.monitor_readability(m_listen_socket);
m_selector.monitor_errors(m_listen_socket);
}
}
void session_impl::operator()()
{
eh_initializer();
@ -304,41 +366,8 @@ namespace libtorrent
try
{
#endif
// create listener socket
// TODO: if this fails, no alert is generated
boost::shared_ptr<socket> listener(new socket(socket::tcp, false));
for(;;)
{
try
{
listener->listen(m_listen_interface, 10);
}
catch (network_error& e)
{
if (e.error_code() == socket::address_not_available)
{
std::string msg = "cannot listen on the given interface '" + m_listen_interface.as_string() + "'";
m_alerts.post_alert(listen_failed_alert(msg));
return;
}
m_listen_interface.port++;
if (m_listen_interface.port > m_listen_port_range.second)
{
m_alerts.post_alert(listen_failed_alert(
"none of the ports in the given range could be opened"));
return;
}
continue;
}
break;
}
#ifndef NDEBUG
(*m_logger) << "listening on port: " << m_listen_interface.port << "\n";
#endif
m_selector.monitor_readability(listener);
m_selector.monitor_errors(listener);
open_listen_port();
std::vector<boost::shared_ptr<socket> > readable_clients;
std::vector<boost::shared_ptr<socket> > writable_clients;
@ -470,9 +499,10 @@ namespace libtorrent
i != readable_clients.end();
++i)
{
// special case for listener socket
if (*i == listener)
// special case for m_listen_socket socket
if (*i == m_listen_socket)
{
assert(m_listen_socket);
boost::shared_ptr<libtorrent::socket> s = (*i)->accept();
if (s)
{
@ -747,7 +777,10 @@ namespace libtorrent
}
#endif
}
}}
namespace libtorrent
{
session::session(
std::pair<int, int> listen_port_range
@ -861,6 +894,36 @@ namespace libtorrent
}
}
bool session::listen_on(
std::pair<int, int> const& port_range
, const char* net_interface)
{
boost::mutex::scoped_lock l(m_impl.m_mutex);
if (m_impl.m_listen_socket)
{
m_impl.m_selector.remove(m_impl.m_listen_socket);
m_impl.m_listen_socket.reset();
}
m_impl.m_listen_port_range = port_range;
m_impl.m_listen_interface = address(net_interface, port_range.first);
m_impl.open_listen_port();
return m_impl.m_listen_socket;
}
unsigned short session::listen_port() const
{
boost::mutex::scoped_lock l(m_impl.m_mutex);
return m_impl.m_listen_interface.port;
}
bool session::is_listening() const
{
boost::mutex::scoped_lock l(m_impl.m_mutex);
return m_impl.m_listen_socket;
}
void session::set_http_settings(const http_settings& s)
{
boost::mutex::scoped_lock l(m_impl.m_mutex);