0
0
mirror of https://github.com/OpenVPN/openvpn3.git synced 2024-09-20 20:13:05 +02:00
openvpn3/openvpn/transport/udplink.hpp
Samuli Seppänen 04b2a3c9b7 Switch from AGPLv3 to GPLv3
Signed-off-by: Samuli Seppänen <samuli@openvpn.net>
2017-03-16 14:43:55 +02:00

247 lines
6.4 KiB
C++

// OpenVPN -- An application to securely tunnel IP networks
// over a single port, with support for SSL/TLS-based
// session authentication and key exchange,
// packet encryption, packet authentication, and
// packet compression.
//
// Copyright (C) 2012-2016 OpenVPN Technologies, Inc.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License Version 3
// as published by the Free Software Foundation.
//
// 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 in the COPYING file.
// If not, see <http://www.gnu.org/licenses/>.
// Low-level UDP transport object.
#ifndef OPENVPN_TRANSPORT_UDPLINK_H
#define OPENVPN_TRANSPORT_UDPLINK_H
#include <memory>
#include <asio.hpp>
#include <openvpn/common/size.hpp>
#include <openvpn/common/rc.hpp>
#include <openvpn/frame/frame.hpp>
#include <openvpn/log/sessionstats.hpp>
#ifdef OPENVPN_GREMLIN
#include <openvpn/transport/gremlin.hpp>
#endif
#if defined(OPENVPN_DEBUG_UDPLINK) && OPENVPN_DEBUG_UDPLINK >= 1
#define OPENVPN_LOG_UDPLINK_ERROR(x) OPENVPN_LOG(x)
#else
#define OPENVPN_LOG_UDPLINK_ERROR(x)
#endif
#if defined(OPENVPN_DEBUG_UDPLINK) && OPENVPN_DEBUG_UDPLINK >= 3
#define OPENVPN_LOG_UDPLINK_VERBOSE(x) OPENVPN_LOG(x)
#else
#define OPENVPN_LOG_UDPLINK_VERBOSE(x)
#endif
namespace openvpn {
namespace UDPTransport {
typedef asio::ip::udp::endpoint AsioEndpoint;
enum {
SEND_SOCKET_HALTED=-1,
SEND_PARTIAL=-2,
};
struct PacketFrom
{
typedef std::unique_ptr<PacketFrom> SPtr;
BufferAllocated buf;
AsioEndpoint sender_endpoint;
};
template <typename ReadHandler>
class Link : public RC<thread_unsafe_refcount>
{
public:
typedef RCPtr<Link> Ptr;
Link(ReadHandler read_handler_arg,
asio::ip::udp::socket& socket_arg,
const Frame::Context& frame_context_arg,
const SessionStats::Ptr& stats_arg)
: socket(socket_arg),
halt(false),
read_handler(read_handler_arg),
frame_context(frame_context_arg),
stats(stats_arg)
{
}
#ifdef OPENVPN_GREMLIN
void gremlin_config(const Gremlin::Config::Ptr& config)
{
if (config)
gremlin.reset(new Gremlin::SendRecvQueue(socket.get_executor().context(), config, false));
}
#endif
// Returns 0 on success, or a system error code on error.
// May also return SEND_PARTIAL or SEND_SOCKET_HALTED.
int send(const Buffer& buf, const AsioEndpoint* endpoint)
{
#ifdef OPENVPN_GREMLIN
if (gremlin)
{
gremlin_send(buf, endpoint);
return 0;
}
else
#endif
return do_send(buf, endpoint);
}
void start(const int n_parallel)
{
if (!halt)
{
for (int i = 0; i < n_parallel; i++)
queue_read(nullptr);
}
}
void stop()
{
halt = true;
#ifdef OPENVPN_GREMLIN
if (gremlin)
gremlin->stop();
#endif
}
void reset_align_adjust(const size_t align_adjust)
{
frame_context.reset_align_adjust(align_adjust);
}
~Link() { stop(); }
private:
void queue_read(PacketFrom *udpfrom)
{
OPENVPN_LOG_UDPLINK_VERBOSE("UDPLink::queue_read");
if (!udpfrom)
udpfrom = new PacketFrom();
frame_context.prepare(udpfrom->buf);
socket.async_receive_from(frame_context.mutable_buffers_1(udpfrom->buf),
udpfrom->sender_endpoint,
[self=Ptr(this), udpfrom](const asio::error_code& error, const size_t bytes_recvd)
{
self->handle_read(udpfrom, error, bytes_recvd);
});
}
void handle_read(PacketFrom *udpfrom, const asio::error_code& error, const size_t bytes_recvd)
{
OPENVPN_LOG_UDPLINK_VERBOSE("UDPLink::handle_read: " << error.message());
PacketFrom::SPtr pfp(udpfrom);
if (!halt)
{
if (bytes_recvd)
{
if (!error)
{
OPENVPN_LOG_UDPLINK_VERBOSE("UDP[" << bytes_recvd << "] from " << pfp->sender_endpoint);
pfp->buf.set_size(bytes_recvd);
stats->inc_stat(SessionStats::BYTES_IN, bytes_recvd);
stats->inc_stat(SessionStats::PACKETS_IN, 1);
#ifdef OPENVPN_GREMLIN
if (gremlin)
gremlin_recv(pfp);
else
#endif
read_handler->udp_read_handler(pfp);
}
else
{
OPENVPN_LOG_UDPLINK_ERROR("UDP recv error: " << error.message());
stats->error(Error::NETWORK_RECV_ERROR);
}
}
if (!halt)
queue_read(pfp.release()); // reuse PacketFrom object if still available
}
}
int do_send(const Buffer& buf, const AsioEndpoint* endpoint)
{
if (!halt)
{
try {
const size_t wrote = endpoint
? socket.send_to(buf.const_buffers_1(), *endpoint)
: socket.send(buf.const_buffers_1());
stats->inc_stat(SessionStats::BYTES_OUT, wrote);
stats->inc_stat(SessionStats::PACKETS_OUT, 1);
if (wrote == buf.size())
return 0;
else
{
OPENVPN_LOG_UDPLINK_ERROR("UDP partial send error");
stats->error(Error::NETWORK_SEND_ERROR);
return SEND_PARTIAL;
}
}
catch (asio::system_error& e)
{
OPENVPN_LOG_UDPLINK_ERROR("UDP send error: " << e.what());
stats->error(Error::NETWORK_SEND_ERROR);
return e.code().value();
}
}
else
return SEND_SOCKET_HALTED;
}
#ifdef OPENVPN_GREMLIN
void gremlin_send(const Buffer& buf, const AsioEndpoint* endpoint)
{
std::unique_ptr<AsioEndpoint> ep;
if (endpoint)
ep.reset(new AsioEndpoint(*endpoint));
gremlin->send_queue([self=Ptr(this), buf=BufferAllocated(buf, 0), ep=std::move(ep)]() mutable {
if (!self->halt)
self->do_send(buf, ep.get());
});
}
void gremlin_recv(PacketFrom::SPtr& pfp)
{
gremlin->recv_queue([self=Ptr(this), pfp=std::move(pfp)]() mutable {
if (!self->halt)
self->read_handler->udp_read_handler(pfp);
});
}
#endif
asio::ip::udp::socket& socket;
bool halt;
ReadHandler read_handler;
Frame::Context frame_context;
SessionStats::Ptr stats;
#ifdef OPENVPN_GREMLIN
std::unique_ptr<Gremlin::SendRecvQueue> gremlin;
#endif
};
}
} // namespace openvpn
#endif