mirror of
https://github.com/transmission/transmission
synced 2025-01-05 06:23:11 +00:00
3b03494580
* refactor: use BufferReader, BufferWriter in peer-socket * feat: expose GrowthFactor in tr-buffer * perf: choose better defaults for the peer message buffers * chore: sync tests * refactor: use small::map in ActiveRequests::Impl
296 lines
7 KiB
C++
296 lines
7 KiB
C++
// This file Copyright © 2022-2023 Mnemosyne LLC.
|
|
// It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only),
|
|
// or any future license endorsed by Mnemosyne LLC.
|
|
// License text can be found in the licenses/ folder.
|
|
|
|
#pragma once
|
|
|
|
#include <algorithm> // for std::copy_n
|
|
#include <cstddef>
|
|
#include <iterator>
|
|
#include <limits>
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <string>
|
|
#include <string_view>
|
|
|
|
#include <small/vector.hpp>
|
|
|
|
#include "error.h"
|
|
#include "net.h" // tr_socket_t
|
|
#include "tr-assert.h"
|
|
#include "utils-ev.h"
|
|
#include "utils.h" // for tr_htonll(), tr_ntohll()
|
|
|
|
namespace libtransmission
|
|
{
|
|
|
|
template<typename value_type>
|
|
class BufferReader
|
|
{
|
|
public:
|
|
virtual ~BufferReader() = default;
|
|
virtual void drain(size_t n_bytes) = 0;
|
|
[[nodiscard]] virtual size_t size() const noexcept = 0;
|
|
[[nodiscard]] virtual value_type const* data() const = 0;
|
|
|
|
[[nodiscard]] auto empty() const noexcept
|
|
{
|
|
return size() == 0;
|
|
}
|
|
|
|
[[nodiscard]] auto const* begin() const noexcept
|
|
{
|
|
return data();
|
|
}
|
|
|
|
[[nodiscard]] auto const* end() const noexcept
|
|
{
|
|
return begin() + size();
|
|
}
|
|
|
|
template<typename T>
|
|
[[nodiscard]] TR_CONSTEXPR20 bool starts_with(T const& needle) const
|
|
{
|
|
auto const n_bytes = std::size(needle);
|
|
auto const needle_begin = reinterpret_cast<value_type const*>(std::data(needle));
|
|
auto const needle_end = needle_begin + n_bytes;
|
|
return n_bytes <= size() && std::equal(needle_begin, needle_end, data());
|
|
}
|
|
|
|
[[nodiscard]] auto to_string_view() const
|
|
{
|
|
return std::string_view{ reinterpret_cast<char const*>(data()), size() };
|
|
}
|
|
|
|
[[nodiscard]] auto to_string() const
|
|
{
|
|
return std::string{ to_string_view() };
|
|
}
|
|
|
|
void to_buf(void* tgt, size_t n_bytes)
|
|
{
|
|
n_bytes = std::min(n_bytes, size());
|
|
std::copy_n(data(), n_bytes, static_cast<value_type*>(tgt));
|
|
drain(n_bytes);
|
|
}
|
|
|
|
[[nodiscard]] auto to_uint8()
|
|
{
|
|
auto tmp = uint8_t{};
|
|
to_buf(&tmp, sizeof(tmp));
|
|
return tmp;
|
|
}
|
|
|
|
[[nodiscard]] uint16_t to_uint16()
|
|
{
|
|
auto tmp = uint16_t{};
|
|
to_buf(&tmp, sizeof(tmp));
|
|
return ntohs(tmp);
|
|
}
|
|
|
|
[[nodiscard]] uint32_t to_uint32()
|
|
{
|
|
auto tmp = uint32_t{};
|
|
to_buf(&tmp, sizeof(tmp));
|
|
return ntohl(tmp);
|
|
}
|
|
|
|
[[nodiscard]] uint64_t to_uint64()
|
|
{
|
|
auto tmp = uint64_t{};
|
|
to_buf(&tmp, sizeof(tmp));
|
|
return tr_ntohll(tmp);
|
|
}
|
|
|
|
// Returns the number of bytes written. Check `error` for error.
|
|
size_t to_socket(tr_socket_t sockfd, size_t n_bytes, tr_error** error = nullptr)
|
|
{
|
|
n_bytes = std::min(n_bytes, size());
|
|
|
|
if (n_bytes == 0U)
|
|
{
|
|
return {};
|
|
}
|
|
|
|
if (auto const n_sent = send(sockfd, reinterpret_cast<char const*>(data()), n_bytes, 0); n_sent >= 0U)
|
|
{
|
|
drain(n_sent);
|
|
return n_sent;
|
|
}
|
|
|
|
auto const err = sockerrno;
|
|
tr_error_set(error, err, tr_net_strerror(err));
|
|
return {};
|
|
}
|
|
|
|
void clear()
|
|
{
|
|
drain(size());
|
|
}
|
|
};
|
|
|
|
template<typename value_type>
|
|
class BufferWriter
|
|
{
|
|
public:
|
|
virtual ~BufferWriter() = default;
|
|
virtual std::pair<value_type*, size_t> reserve_space(size_t n_bytes) = 0;
|
|
virtual void commit_space(size_t n_bytes) = 0;
|
|
|
|
void add(void const* span_begin, size_t span_len)
|
|
{
|
|
auto [buf, buflen] = reserve_space(span_len);
|
|
std::copy_n(reinterpret_cast<value_type const*>(span_begin), span_len, buf);
|
|
commit_space(span_len);
|
|
}
|
|
|
|
template<typename ContiguousContainer>
|
|
void add(ContiguousContainer const& container)
|
|
{
|
|
add(std::data(container), std::size(container));
|
|
}
|
|
|
|
template<typename OneByteType>
|
|
void push_back(OneByteType ch)
|
|
{
|
|
add(&ch, 1);
|
|
}
|
|
|
|
void add_uint8(uint8_t uch)
|
|
{
|
|
add(&uch, 1);
|
|
}
|
|
|
|
void add_uint16(uint16_t hs)
|
|
{
|
|
uint16_t const ns = htons(hs);
|
|
add(&ns, sizeof(ns));
|
|
}
|
|
|
|
void add_hton16(uint16_t hs)
|
|
{
|
|
add_uint16(hs);
|
|
}
|
|
|
|
void add_uint32(uint32_t hl)
|
|
{
|
|
uint32_t const nl = htonl(hl);
|
|
add(&nl, sizeof(nl));
|
|
}
|
|
|
|
void eadd_hton32(uint32_t hl)
|
|
{
|
|
add_uint32(hl);
|
|
}
|
|
|
|
void add_uint64(uint64_t hll)
|
|
{
|
|
uint64_t const nll = tr_htonll(hll);
|
|
add(&nll, sizeof(nll));
|
|
}
|
|
|
|
void add_hton64(uint64_t hll)
|
|
{
|
|
add_uint64(hll);
|
|
}
|
|
|
|
void add_port(tr_port port)
|
|
{
|
|
auto nport = port.network();
|
|
add(&nport, sizeof(nport));
|
|
}
|
|
|
|
size_t add_socket(tr_socket_t sockfd, size_t n_bytes, tr_error** error = nullptr)
|
|
{
|
|
if (n_bytes == 0U)
|
|
{
|
|
return {};
|
|
}
|
|
|
|
auto const [buf, buflen] = reserve_space(n_bytes);
|
|
auto const n_read = recv(sockfd, reinterpret_cast<char*>(buf), std::min(n_bytes, buflen), 0);
|
|
auto const err = sockerrno;
|
|
|
|
if (n_read > 0)
|
|
{
|
|
commit_space(n_read);
|
|
return static_cast<size_t>(n_read);
|
|
}
|
|
|
|
// When a stream socket peer has performed an orderly shutdown,
|
|
// the return value will be 0 (the traditional "end-of-file" return).
|
|
if (n_read == 0)
|
|
{
|
|
tr_error_set_from_errno(error, ENOTCONN);
|
|
}
|
|
else
|
|
{
|
|
tr_error_set(error, err, tr_net_strerror(err));
|
|
}
|
|
|
|
return {};
|
|
}
|
|
};
|
|
|
|
template<size_t N, typename value_type = std::byte, typename GrowthFactor = std::ratio<3, 2>>
|
|
class SmallBuffer final
|
|
: public BufferReader<value_type>
|
|
, public BufferWriter<value_type>
|
|
{
|
|
public:
|
|
SmallBuffer() = default;
|
|
SmallBuffer(SmallBuffer&&) = delete;
|
|
SmallBuffer& operator=(SmallBuffer&&) = delete;
|
|
SmallBuffer(SmallBuffer const&) = delete;
|
|
SmallBuffer& operator=(SmallBuffer const&) = delete;
|
|
|
|
explicit SmallBuffer(void const* const data, size_t n_bytes)
|
|
{
|
|
BufferWriter<value_type>::add(data, n_bytes);
|
|
}
|
|
|
|
template<typename ContiguousContainer>
|
|
explicit SmallBuffer(ContiguousContainer const& data)
|
|
: SmallBuffer{ std::data(data), std::size(data) }
|
|
{
|
|
}
|
|
|
|
[[nodiscard]] size_t size() const noexcept override
|
|
{
|
|
return end_pos_ - begin_pos_;
|
|
}
|
|
|
|
[[nodiscard]] value_type const* data() const override
|
|
{
|
|
return std::data(buf_) + begin_pos_;
|
|
}
|
|
|
|
void drain(size_t n_bytes) override
|
|
{
|
|
begin_pos_ += std::min(n_bytes, size());
|
|
|
|
if (begin_pos_ == end_pos_)
|
|
{
|
|
begin_pos_ = end_pos_ = 0U;
|
|
}
|
|
}
|
|
|
|
virtual std::pair<value_type*, size_t> reserve_space(size_t n_bytes) override
|
|
{
|
|
buf_.resize(end_pos_ + n_bytes);
|
|
return { &buf_[end_pos_], n_bytes };
|
|
}
|
|
|
|
virtual void commit_space(size_t n_bytes) override
|
|
{
|
|
end_pos_ += n_bytes;
|
|
}
|
|
|
|
private:
|
|
small::vector<value_type, N, std::allocator<value_type>, std::true_type, size_t, GrowthFactor> buf_ = {};
|
|
size_t begin_pos_ = {};
|
|
size_t end_pos_ = {};
|
|
};
|
|
|
|
} // namespace libtransmission
|