Telegram-Android/TMessagesProj/jni/voip/tgcalls/v2/SignalingSctpConnection.cpp
2023-02-19 01:24:25 +04:00

189 lines
6.5 KiB
C++

#include "v2/SignalingSctpConnection.h"
#include <random>
#include "rtc_base/async_tcp_socket.h"
#include "p2p/base/basic_packet_socket_factory.h"
#include "rtc_base/logging.h"
#include "p2p/base/packet_transport_internal.h"
#include "media/sctp/sctp_transport_factory.h"
#include "FieldTrialsConfig.h"
namespace tgcalls {
class SignalingPacketTransport : public rtc::PacketTransportInternal {
public:
SignalingPacketTransport(std::shared_ptr<Threads> threads, std::function<void(const std::vector<uint8_t> &)> emitData) :
_threads(threads),
_emitData(emitData),
_transportName("signaling") {
}
virtual ~SignalingPacketTransport() {
}
void receiveData(std::vector<uint8_t> const &data) {
RTC_LOG(LS_INFO) << "SignalingPacketTransport: adding data of " << data.size() << " bytes";
SignalReadPacket.emit(this, (const char *)data.data(), data.size(), -1, 0);
}
virtual const std::string& transport_name() const override {
return _transportName;
}
virtual bool writable() const override {
return true;
}
virtual bool receiving() const override {
return false;
}
// Attempts to send the given packet.
// The return value is < 0 on failure. The return value in failure case is not
// descriptive. Depending on failure cause and implementation details
// GetError() returns an descriptive errno.h error value.
// This mimics posix socket send() or sendto() behavior.
// TODO(johan): Reliable, meaningful, consistent error codes for all
// implementations would be nice.
// TODO(johan): Remove the default argument once channel code is updated.
virtual int SendPacket(const char* data,
size_t len,
const rtc::PacketOptions& options,
int flags = 0) override {
_emitData(std::vector<uint8_t>(data, data + len));
rtc::SentPacket sentPacket;
sentPacket.packet_id = options.packet_id;
SignalSentPacket.emit(this, sentPacket);
return (int)len;
}
virtual int SetOption(rtc::Socket::Option opt, int value) override {
return 0;
}
virtual bool GetOption(rtc::Socket::Option opt, int* value) override {
return 0;
}
virtual int GetError() override {
return 0;
}
virtual absl::optional<rtc::NetworkRoute> network_route() const override {
return absl::nullopt;
}
private:
std::shared_ptr<Threads> _threads;
std::function<void(const std::vector<uint8_t> &)> _onIncomingData;
std::function<void(const std::vector<uint8_t> &)> _emitData;
std::string _transportName;
};
SignalingSctpConnection::SignalingSctpConnection(std::shared_ptr<Threads> threads, std::function<void(const std::vector<uint8_t> &)> onIncomingData, std::function<void(const std::vector<uint8_t> &)> emitData) :
_threads(threads),
_emitData(emitData),
_onIncomingData(onIncomingData) {
_threads->getNetworkThread()->BlockingCall([&]() {
_packetTransport = std::make_unique<SignalingPacketTransport>(threads, emitData);
_sctpTransportFactory.reset(new cricket::SctpTransportFactory(_threads->getNetworkThread()));
_sctpTransport = _sctpTransportFactory->CreateSctpTransport(_packetTransport.get());
_sctpTransport->OpenStream(0);
_sctpTransport->SetDataChannelSink(this);
// TODO: should we disconnect the data channel sink?
_sctpTransport->Start(5000, 5000, 262144);
});
}
void SignalingSctpConnection::OnReadyToSend() {
assert(_threads->getNetworkThread()->IsCurrent());
_isReadyToSend = true;
auto pendingData = _pendingData;
_pendingData.clear();
for (const auto &data : pendingData) {
webrtc::SendDataParams params;
params.type = webrtc::DataMessageType::kBinary;
params.ordered = true;
rtc::CopyOnWriteBuffer payload;
payload.AppendData(data.data(), data.size());
cricket::SendDataResult result;
_sctpTransport->SendData(0, params, payload, &result);
if (result == cricket::SendDataResult::SDR_SUCCESS) {
RTC_LOG(LS_INFO) << "SignalingSctpConnection: sent data of " << data.size() << " bytes";
} else {
_isReadyToSend = false;
_pendingData.push_back(data);
RTC_LOG(LS_INFO) << "SignalingSctpConnection: send error, storing data until ready to send (" << _pendingData.size() << " items)";
}
}
}
void SignalingSctpConnection::OnTransportClosed(webrtc::RTCError error) {
assert(_threads->getNetworkThread()->IsCurrent());
}
void SignalingSctpConnection::OnDataReceived(int channel_id, webrtc::DataMessageType type, const rtc::CopyOnWriteBuffer& buffer) {
assert(_threads->getNetworkThread()->IsCurrent());
_onIncomingData(std::vector<uint8_t>(buffer.data(), buffer.data() + buffer.size()));
}
SignalingSctpConnection::~SignalingSctpConnection() {
_threads->getNetworkThread()->BlockingCall([&]() {
_sctpTransport.reset();
_sctpTransportFactory.reset();
_packetTransport.reset();
});
}
void SignalingSctpConnection::start() {
}
void SignalingSctpConnection::receiveExternal(const std::vector<uint8_t> &data) {
_threads->getNetworkThread()->BlockingCall([&]() {
_packetTransport->receiveData(data);
});
}
void SignalingSctpConnection::send(const std::vector<uint8_t> &data) {
_threads->getNetworkThread()->BlockingCall([&]() {
if (_isReadyToSend) {
webrtc::SendDataParams params;
params.type = webrtc::DataMessageType::kBinary;
params.ordered = true;
rtc::CopyOnWriteBuffer payload;
payload.AppendData(data.data(), data.size());
cricket::SendDataResult result;
_sctpTransport->SendData(0, params, payload, &result);
if (result == cricket::SendDataResult::SDR_ERROR) {
_isReadyToSend = false;
_pendingData.push_back(data);
RTC_LOG(LS_INFO) << "SignalingSctpConnection: send error, storing data until ready to send (" << _pendingData.size() << " items)";
} else {
RTC_LOG(LS_INFO) << "SignalingSctpConnection: sent data of " << data.size() << " bytes";
}
} else {
_pendingData.push_back(data);
RTC_LOG(LS_INFO) << "SignalingSctpConnection: not ready to send, storing data until ready to send (" << _pendingData.size() << " items)";
}
});
}
}