Repo created
This commit is contained in:
parent
81b91f4139
commit
f8c34fa5ee
22732 changed files with 4815320 additions and 2 deletions
187
TMessagesProj/jni/voip/tgcalls/v2/SignalingSctpConnection.cpp
Normal file
187
TMessagesProj/jni/voip/tgcalls/v2/SignalingSctpConnection.cpp
Normal file
|
|
@ -0,0 +1,187 @@
|
|||
#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());
|
||||
|
||||
webrtc::RTCError sendError = _sctpTransport->SendData(0, params, payload);
|
||||
|
||||
if (sendError.ok()) {
|
||||
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());
|
||||
|
||||
webrtc::RTCError sendError = _sctpTransport->SendData(0, params, payload);
|
||||
|
||||
if (!sendError.ok()) {
|
||||
_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)";
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue