blob: 3ceed0a011e7b546faddf48bbda211dfdd524c55 [file] [log] [blame]
/*
* Copyright (C) 2017 Apple Inc.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "LibWebRTCDataChannelHandler.h"
#if USE(LIBWEBRTC)
#include "EventNames.h"
#include "LibWebRTCUtils.h"
#include "RTCDataChannel.h"
#include "RTCError.h"
#include <wtf/MainThread.h>
namespace WebCore {
webrtc::DataChannelInit LibWebRTCDataChannelHandler::fromRTCDataChannelInit(const RTCDataChannelInit& options)
{
webrtc::DataChannelInit init;
if (options.ordered)
init.ordered = *options.ordered;
if (options.maxPacketLifeTime)
init.maxRetransmitTime = *options.maxPacketLifeTime;
if (options.maxRetransmits)
init.maxRetransmits = *options.maxRetransmits;
init.protocol = options.protocol.utf8().data();
if (options.negotiated)
init.negotiated = *options.negotiated;
if (options.id)
init.id = *options.id;
init.priority = fromRTCPriorityType(options.priority);
return init;
}
LibWebRTCDataChannelHandler::LibWebRTCDataChannelHandler(rtc::scoped_refptr<webrtc::DataChannelInterface>&& channel)
: m_channel(WTFMove(channel))
{
ASSERT(m_channel);
checkState();
m_channel->RegisterObserver(this);
}
LibWebRTCDataChannelHandler::~LibWebRTCDataChannelHandler()
{
m_channel->UnregisterObserver();
}
RTCDataChannelInit LibWebRTCDataChannelHandler::dataChannelInit() const
{
auto protocol = m_channel->protocol();
auto label = m_channel->label();
RTCDataChannelInit init;
init.ordered = m_channel->ordered();
init.maxPacketLifeTime = m_channel->maxRetransmitTime();
init.maxRetransmits = m_channel->maxRetransmits();
init.protocol = fromStdString(protocol);
init.negotiated = m_channel->negotiated();
init.id = m_channel->id();
init.priority = toRTCPriorityType(m_channel->priority());
return init;
}
String LibWebRTCDataChannelHandler::label() const
{
return fromStdString(m_channel->label());
}
void LibWebRTCDataChannelHandler::setClient(RTCDataChannelHandlerClient& client, ScriptExecutionContextIdentifier contextIdentifier)
{
Locker locker { m_clientLock };
ASSERT(!m_client);
ASSERT(!m_hasClient);
m_hasClient = true;
m_client = client;
m_contextIdentifier = contextIdentifier;
for (auto& message : m_bufferedMessages) {
switchOn(message, [&](Ref<FragmentedSharedBuffer>& data) {
client.didReceiveRawData(data->makeContiguous()->data(), data->size());
}, [&](String& text) {
client.didReceiveStringData(text);
}, [&](StateChange stateChange) {
if (stateChange.error) {
if (auto rtcError = toRTCError(*stateChange.error))
client.didDetectError(rtcError.releaseNonNull());
}
client.didChangeReadyState(stateChange.state);
});
}
m_bufferedMessages.clear();
}
bool LibWebRTCDataChannelHandler::sendStringData(const CString& utf8Text)
{
return m_channel->Send({ rtc::CopyOnWriteBuffer(utf8Text.data(), utf8Text.length()), false });
}
bool LibWebRTCDataChannelHandler::sendRawData(const uint8_t* data, size_t length)
{
return m_channel->Send({ rtc::CopyOnWriteBuffer(data, length), true });
}
void LibWebRTCDataChannelHandler::close()
{
m_channel->Close();
}
void LibWebRTCDataChannelHandler::OnStateChange()
{
checkState();
}
void LibWebRTCDataChannelHandler::checkState()
{
std::optional<webrtc::RTCError> error;
RTCDataChannelState state;
switch (m_channel->state()) {
case webrtc::DataChannelInterface::kConnecting:
state = RTCDataChannelState::Connecting;
break;
case webrtc::DataChannelInterface::kOpen:
state = RTCDataChannelState::Open;
break;
case webrtc::DataChannelInterface::kClosing:
state = RTCDataChannelState::Closing;
break;
case webrtc::DataChannelInterface::kClosed:
error = m_channel->error();
state = RTCDataChannelState::Closed;
break;
}
Locker locker { m_clientLock };
if (!m_hasClient) {
m_bufferedMessages.append(StateChange { state, WTFMove(error) });
return;
}
postTask([client = m_client, state, error = WTFMove(error)] {
if (!client)
return;
if (error && !error->ok()) {
auto rtcError = toRTCError(*error);
if (!rtcError)
rtcError = RTCError::create(RTCError::Init { RTCErrorDetailType::DataChannelFailure, { }, { }, { }, { } }, String { });
client->didDetectError(rtcError.releaseNonNull());
}
client->didChangeReadyState(state);
});
}
void LibWebRTCDataChannelHandler::OnMessage(const webrtc::DataBuffer& buffer)
{
Locker locker { m_clientLock };
if (!m_hasClient) {
auto* data = buffer.data.data<uint8_t>();
if (buffer.binary)
m_bufferedMessages.append(SharedBuffer::create(data, buffer.size()));
else
m_bufferedMessages.append(String::fromUTF8(data, buffer.size()));
return;
}
std::unique_ptr<webrtc::DataBuffer> protectedBuffer(new webrtc::DataBuffer(buffer));
postTask([client = m_client, buffer = WTFMove(protectedBuffer)] {
if (!client)
return;
auto* data = buffer->data.data<uint8_t>();
if (buffer->binary)
client->didReceiveRawData(data, buffer->size());
else
client->didReceiveStringData(String::fromUTF8(data, buffer->size()));
});
}
void LibWebRTCDataChannelHandler::OnBufferedAmountChange(uint64_t amount)
{
Locker locker { m_clientLock };
if (!m_hasClient)
return;
postTask([client = m_client, amount] {
if (client)
client->bufferedAmountIsDecreasing(static_cast<size_t>(amount));
});
}
void LibWebRTCDataChannelHandler::postTask(Function<void()>&& function)
{
ASSERT(m_clientLock.isHeld());
if (!m_contextIdentifier) {
callOnMainThread(WTFMove(function));
return;
}
ScriptExecutionContext::postTaskTo(m_contextIdentifier, WTFMove(function));
}
} // namespace WebCore
#endif // USE(LIBWEBRTC)