0
0
mirror of https://github.com/OpenVPN/openvpn3.git synced 2024-09-20 20:13:05 +02:00
openvpn3/openvpn/frame/memq_stream.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

111 lines
3.1 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/>.
// A queue of buffers for handling streamed data such as data received
// from or to be sent to a TCP socket
#ifndef OPENVPN_FRAME_MEMQ_STREAM_H
#define OPENVPN_FRAME_MEMQ_STREAM_H
#include <algorithm>
#include <openvpn/common/size.hpp>
#include <openvpn/common/exception.hpp>
#include <openvpn/buffer/memq.hpp>
#include <openvpn/frame/frame.hpp>
namespace openvpn {
class MemQStream : public MemQBase {
public:
OPENVPN_SIMPLE_EXCEPTION(frame_uninitialized);
MemQStream() {}
explicit MemQStream(const Frame::Ptr& frame) : frame_(frame) {}
void set_frame(const Frame::Ptr& frame) { frame_ = frame; }
size_t pending() const
{
return total_length();
}
void write(const unsigned char *data, size_t size)
{
if (frame_)
{
const Frame::Context& fc = (*frame_)[Frame::READ_BIO_MEMQ_STREAM];
if (size)
{
ConstBuffer b(data, size, true);
// Any residual space remaining in most recently pushed buffer?
if (!q.empty())
{
BufferPtr& qb = q.back();
const size_t write_size = std::min(b.size(), fc.remaining_payload(*qb));
const unsigned char *from = b.read_alloc(write_size);
qb->write(from, write_size);
length += write_size;
}
// Start a new buffer
while (b.size())
{
BufferPtr newbuf(new BufferAllocated);
fc.prepare(*newbuf);
const size_t write_size = std::min(b.size(), fc.payload());
const unsigned char *from = b.read_alloc(write_size);
newbuf->write(from, write_size);
q.push_back(newbuf);
length += write_size;
}
}
}
else
throw frame_uninitialized();
}
size_t read(unsigned char *data, size_t len)
{
Buffer b(data, len, false);
while (!q.empty())
{
const size_t remaining = b.remaining();
if (!remaining)
break;
BufferPtr& qf = q.front();
const size_t read_size = std::min(remaining, qf->size());
unsigned char *to = b.write_alloc(read_size);
qf->read(to, read_size);
length -= read_size;
if (qf->empty())
q.pop_front();
}
return b.size();
}
private:
Frame::Ptr frame_;
};
} // namespace openvpn
#endif // OPENVPN_FRAME_MEMQ_STREAM_H