2003-10-23 01:00:57 +02:00
|
|
|
/*
|
|
|
|
|
|
|
|
Copyright (c) 2003, Arvid Norberg
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions
|
|
|
|
are met:
|
|
|
|
|
|
|
|
* Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
* Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in
|
|
|
|
the documentation and/or other materials provided with the distribution.
|
|
|
|
* Neither the name of the author nor the names of its
|
|
|
|
contributors may be used to endorse or promote products derived
|
|
|
|
from this software without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2004-01-01 22:25:59 +01:00
|
|
|
#ifndef TORRENT_SOCKET_HPP_INCLUDED
|
|
|
|
#define TORRENT_SOCKET_HPP_INCLUDED
|
2003-10-23 01:00:57 +02:00
|
|
|
|
2004-01-25 19:18:36 +01:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(push, 1)
|
|
|
|
#endif
|
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
// if building as Objective C++, asio's template
|
|
|
|
// parameters Protocol has to be renamed to avoid
|
|
|
|
// colliding with keywords
|
2003-10-23 01:00:57 +02:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
#ifdef __OBJC__
|
|
|
|
#define Protocol Protocol_
|
2004-01-25 19:18:36 +01:00
|
|
|
#endif
|
|
|
|
|
2008-05-20 11:45:55 +02:00
|
|
|
#include <boost/version.hpp>
|
|
|
|
|
2008-05-20 08:03:46 +02:00
|
|
|
#if BOOST_VERSION < 103500
|
|
|
|
#include <asio/ip/tcp.hpp>
|
|
|
|
#include <asio/ip/udp.hpp>
|
|
|
|
#include <asio/io_service.hpp>
|
|
|
|
#include <asio/deadline_timer.hpp>
|
|
|
|
#include <asio/write.hpp>
|
|
|
|
#include <asio/read.hpp>
|
|
|
|
#include <asio/time_traits.hpp>
|
|
|
|
#include <asio/basic_deadline_timer.hpp>
|
|
|
|
#else
|
2008-05-03 18:05:42 +02:00
|
|
|
#include <boost/asio/ip/tcp.hpp>
|
|
|
|
#include <boost/asio/ip/udp.hpp>
|
|
|
|
#include <boost/asio/io_service.hpp>
|
|
|
|
#include <boost/asio/deadline_timer.hpp>
|
|
|
|
#include <boost/asio/write.hpp>
|
|
|
|
#include <boost/asio/read.hpp>
|
|
|
|
#include <boost/asio/time_traits.hpp>
|
|
|
|
#include <boost/asio/basic_deadline_timer.hpp>
|
2008-05-20 08:03:46 +02:00
|
|
|
#endif
|
2004-01-25 19:18:36 +01:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
#ifdef __OBJC__
|
|
|
|
#undef Protocol
|
|
|
|
#endif
|
2003-10-23 01:00:57 +02:00
|
|
|
|
2006-08-01 17:27:08 +02:00
|
|
|
#include "libtorrent/io.hpp"
|
2007-04-05 00:27:36 +02:00
|
|
|
#include "libtorrent/time.hpp"
|
2008-07-18 01:41:46 +02:00
|
|
|
#include "libtorrent/error_code.hpp"
|
2009-04-04 11:52:25 +02:00
|
|
|
#include "libtorrent/escape_string.hpp" // for to_string
|
2006-08-01 17:27:08 +02:00
|
|
|
|
2006-04-25 23:04:48 +02:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(pop)
|
|
|
|
#endif
|
2004-01-26 18:39:44 +01:00
|
|
|
|
2003-10-23 01:00:57 +02:00
|
|
|
namespace libtorrent
|
|
|
|
{
|
2007-04-05 00:27:36 +02:00
|
|
|
|
2008-05-20 08:03:46 +02:00
|
|
|
#if BOOST_VERSION < 103500
|
2009-03-06 09:57:00 +01:00
|
|
|
using ::asio::ip::tcp;
|
|
|
|
using ::asio::ip::udp;
|
|
|
|
using ::asio::async_write;
|
|
|
|
using ::asio::async_read;
|
|
|
|
|
|
|
|
typedef ::asio::ip::tcp::socket stream_socket;
|
|
|
|
typedef ::asio::ip::address address;
|
|
|
|
typedef ::asio::ip::address_v4 address_v4;
|
|
|
|
typedef ::asio::ip::address_v6 address_v6;
|
|
|
|
typedef ::asio::ip::udp::socket datagram_socket;
|
|
|
|
typedef ::asio::ip::tcp::acceptor socket_acceptor;
|
|
|
|
typedef ::asio::io_service io_service;
|
|
|
|
typedef ::asio::basic_deadline_timer<libtorrent::ptime> deadline_timer;
|
2008-05-20 08:03:46 +02:00
|
|
|
#else
|
2008-05-03 18:05:42 +02:00
|
|
|
using boost::asio::ip::tcp;
|
|
|
|
using boost::asio::ip::udp;
|
|
|
|
using boost::asio::async_write;
|
|
|
|
using boost::asio::async_read;
|
|
|
|
|
|
|
|
typedef boost::asio::ip::tcp::socket stream_socket;
|
|
|
|
typedef boost::asio::ip::address address;
|
|
|
|
typedef boost::asio::ip::address_v4 address_v4;
|
|
|
|
typedef boost::asio::ip::address_v6 address_v6;
|
|
|
|
typedef boost::asio::ip::udp::socket datagram_socket;
|
|
|
|
typedef boost::asio::ip::tcp::acceptor socket_acceptor;
|
|
|
|
typedef boost::asio::io_service io_service;
|
|
|
|
|
|
|
|
namespace asio = boost::asio;
|
|
|
|
typedef boost::asio::basic_deadline_timer<libtorrent::ptime> deadline_timer;
|
2008-05-20 08:03:46 +02:00
|
|
|
#endif
|
2006-08-01 17:27:08 +02:00
|
|
|
|
2009-04-04 11:52:25 +02:00
|
|
|
inline std::string print_address(address const& addr)
|
|
|
|
{
|
|
|
|
error_code ec;
|
|
|
|
return addr.to_string(ec);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline std::string print_endpoint(tcp::endpoint const& ep)
|
|
|
|
{
|
|
|
|
error_code ec;
|
|
|
|
std::string ret;
|
|
|
|
address const& addr = ep.address();
|
|
|
|
#if TORRENT_USE_IPV6
|
|
|
|
if (addr.is_v6())
|
|
|
|
{
|
|
|
|
ret += '[';
|
|
|
|
ret += addr.to_string(ec);
|
|
|
|
ret += ']';
|
|
|
|
ret += ':';
|
|
|
|
ret += to_string(ep.port()).elems;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
ret += addr.to_string(ec);
|
|
|
|
ret += ':';
|
|
|
|
ret += to_string(ep.port()).elems;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2008-03-29 23:45:55 +01:00
|
|
|
|
2009-04-13 06:22:03 +02:00
|
|
|
inline std::string print_endpoint(udp::endpoint const& ep)
|
|
|
|
{
|
|
|
|
return print_endpoint(tcp::endpoint(ep.address(), ep.port()));
|
|
|
|
}
|
|
|
|
|
2006-08-01 17:27:08 +02:00
|
|
|
namespace detail
|
|
|
|
{
|
2006-09-01 05:06:00 +02:00
|
|
|
template<class OutIt>
|
|
|
|
void write_address(address const& a, OutIt& out)
|
|
|
|
{
|
2009-04-04 18:59:53 +02:00
|
|
|
#if TORRENT_USE_IPV6
|
2006-09-01 05:06:00 +02:00
|
|
|
if (a.is_v4())
|
|
|
|
{
|
2009-04-04 18:59:53 +02:00
|
|
|
#endif
|
2006-09-01 05:06:00 +02:00
|
|
|
write_uint32(a.to_v4().to_ulong(), out);
|
2009-04-04 18:59:53 +02:00
|
|
|
#if TORRENT_USE_IPV6
|
2006-09-01 05:06:00 +02:00
|
|
|
}
|
|
|
|
else if (a.is_v6())
|
|
|
|
{
|
2008-05-03 18:05:42 +02:00
|
|
|
address_v6::bytes_type bytes
|
2006-09-01 05:06:00 +02:00
|
|
|
= a.to_v6().to_bytes();
|
|
|
|
std::copy(bytes.begin(), bytes.end(), out);
|
|
|
|
}
|
2009-04-04 18:59:53 +02:00
|
|
|
#endif
|
2006-09-01 05:06:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<class InIt>
|
|
|
|
address read_v4_address(InIt& in)
|
|
|
|
{
|
|
|
|
unsigned long ip = read_uint32(in);
|
2008-05-03 18:05:42 +02:00
|
|
|
return address_v4(ip);
|
2006-09-01 05:06:00 +02:00
|
|
|
}
|
|
|
|
|
2009-04-04 18:59:53 +02:00
|
|
|
#if TORRENT_USE_IPV6
|
2006-09-01 05:06:00 +02:00
|
|
|
template<class InIt>
|
|
|
|
address read_v6_address(InIt& in)
|
|
|
|
{
|
2008-05-03 18:05:42 +02:00
|
|
|
typedef address_v6::bytes_type bytes_t;
|
2006-09-01 05:06:00 +02:00
|
|
|
bytes_t bytes;
|
|
|
|
for (bytes_t::iterator i = bytes.begin()
|
|
|
|
, end(bytes.end()); i != end; ++i)
|
|
|
|
*i = read_uint8(in);
|
2008-05-03 18:05:42 +02:00
|
|
|
return address_v6(bytes);
|
2006-09-01 05:06:00 +02:00
|
|
|
}
|
2009-04-04 18:59:53 +02:00
|
|
|
#endif
|
2006-09-01 05:06:00 +02:00
|
|
|
|
2006-08-01 17:27:08 +02:00
|
|
|
template<class Endpoint, class OutIt>
|
|
|
|
void write_endpoint(Endpoint const& e, OutIt& out)
|
|
|
|
{
|
2006-09-01 05:06:00 +02:00
|
|
|
write_address(e.address(), out);
|
2006-08-01 17:27:08 +02:00
|
|
|
write_uint16(e.port(), out);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class Endpoint, class InIt>
|
2006-09-01 05:06:00 +02:00
|
|
|
Endpoint read_v4_endpoint(InIt& in)
|
|
|
|
{
|
|
|
|
address addr = read_v4_address(in);
|
|
|
|
int port = read_uint16(in);
|
|
|
|
return Endpoint(addr, port);
|
|
|
|
}
|
|
|
|
|
2009-04-04 18:59:53 +02:00
|
|
|
#if TORRENT_USE_IPV6
|
2006-09-01 05:06:00 +02:00
|
|
|
template<class Endpoint, class InIt>
|
|
|
|
Endpoint read_v6_endpoint(InIt& in)
|
2006-08-01 17:27:08 +02:00
|
|
|
{
|
2006-09-01 05:06:00 +02:00
|
|
|
address addr = read_v6_address(in);
|
2006-08-01 17:27:08 +02:00
|
|
|
int port = read_uint16(in);
|
2006-09-01 05:06:00 +02:00
|
|
|
return Endpoint(addr, port);
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
2009-04-04 18:59:53 +02:00
|
|
|
#endif
|
2006-08-01 17:27:08 +02:00
|
|
|
}
|
2007-11-23 23:14:33 +01:00
|
|
|
|
2009-04-04 18:59:53 +02:00
|
|
|
#if TORRENT_USE_IPV6
|
2007-11-23 23:14:33 +01:00
|
|
|
struct v6only
|
|
|
|
{
|
|
|
|
v6only(bool enable): m_value(enable) {}
|
|
|
|
template<class Protocol>
|
|
|
|
int level(Protocol const&) const { return IPPROTO_IPV6; }
|
|
|
|
template<class Protocol>
|
|
|
|
int name(Protocol const&) const { return IPV6_V6ONLY; }
|
|
|
|
template<class Protocol>
|
|
|
|
int const* data(Protocol const&) const { return &m_value; }
|
|
|
|
template<class Protocol>
|
|
|
|
size_t size(Protocol const&) const { return sizeof(m_value); }
|
|
|
|
int m_value;
|
|
|
|
};
|
2009-04-04 18:59:53 +02:00
|
|
|
#endif
|
2007-11-23 23:14:33 +01:00
|
|
|
|
2009-01-21 02:39:13 +01:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2009-03-08 03:16:35 +01:00
|
|
|
|
|
|
|
#ifndef IPV6_PROTECTION_LEVEL
|
|
|
|
#define IPV6_PROTECTION_LEVEL 30
|
|
|
|
#endif
|
2009-01-21 02:39:13 +01:00
|
|
|
struct v6_protection_level
|
|
|
|
{
|
|
|
|
v6_protection_level(int level): m_value(level) {}
|
|
|
|
template<class Protocol>
|
|
|
|
int level(Protocol const&) const { return IPPROTO_IPV6; }
|
|
|
|
template<class Protocol>
|
|
|
|
int name(Protocol const&) const { return IPV6_PROTECTION_LEVEL; }
|
|
|
|
template<class Protocol>
|
|
|
|
int const* data(Protocol const&) const { return &m_value; }
|
|
|
|
template<class Protocol>
|
|
|
|
size_t size(Protocol const&) const { return sizeof(m_value); }
|
|
|
|
int m_value;
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2008-03-12 17:58:23 +01:00
|
|
|
struct type_of_service
|
|
|
|
{
|
|
|
|
type_of_service(char val): m_value(val) {}
|
|
|
|
template<class Protocol>
|
|
|
|
int level(Protocol const&) const { return IPPROTO_IP; }
|
|
|
|
template<class Protocol>
|
|
|
|
int name(Protocol const&) const { return IP_TOS; }
|
|
|
|
template<class Protocol>
|
|
|
|
char const* data(Protocol const&) const { return &m_value; }
|
|
|
|
template<class Protocol>
|
|
|
|
size_t size(Protocol const&) const { return sizeof(m_value); }
|
|
|
|
char m_value;
|
|
|
|
};
|
2003-10-23 01:00:57 +02:00
|
|
|
}
|
|
|
|
|
2004-01-16 00:57:11 +01:00
|
|
|
#endif // TORRENT_SOCKET_HPP_INCLUDED
|
2004-01-01 22:25:59 +01:00
|
|
|
|