/* * libjingle * Copyright 2013 Google 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. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 THE AUTHOR 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 "talk/media/sctp/sctputils.h" #include "talk/app/webrtc/datachannelinterface.h" #include "talk/base/buffer.h" #include "talk/base/bytebuffer.h" #include "talk/base/logging.h" namespace cricket { // Format defined at // http://tools.ietf.org/html/draft-ietf-rtcweb-data-protocol-00#section-6.1 static const uint8 DATA_CHANNEL_OPEN_MESSAGE_TYPE = 0x03; enum DataChannelOpenMessageChannelType { DCOMCT_ORDERED_RELIABLE = 0x00, DCOMCT_ORDERED_PARTIAL_RTXS = 0x01, DCOMCT_ORDERED_PARTIAL_TIME = 0x02, DCOMCT_UNORDERED_RELIABLE = 0x80, DCOMCT_UNORDERED_PARTIAL_RTXS = 0x81, DCOMCT_UNORDERED_PARTIAL_TIME = 0x82, }; bool ParseDataChannelOpenMessage( const talk_base::Buffer& payload, std::string* label, webrtc::DataChannelInit* config) { // Format defined at // http://tools.ietf.org/html/draft-jesup-rtcweb-data-protocol-04 talk_base::ByteBuffer buffer(payload.data(), payload.length()); uint8 message_type; if (!buffer.ReadUInt8(&message_type)) { LOG(LS_WARNING) << "Could not read OPEN message type."; return false; } if (message_type != DATA_CHANNEL_OPEN_MESSAGE_TYPE) { LOG(LS_WARNING) << "Data Channel OPEN message of unexpected type: " << message_type; return false; } uint8 channel_type; if (!buffer.ReadUInt8(&channel_type)) { LOG(LS_WARNING) << "Could not read OPEN message channel type."; return false; } uint16 priority; if (!buffer.ReadUInt16(&priority)) { LOG(LS_WARNING) << "Could not read OPEN message reliabilility prioirty."; return false; } uint32 reliability_param; if (!buffer.ReadUInt32(&reliability_param)) { LOG(LS_WARNING) << "Could not read OPEN message reliabilility param."; return false; } uint16 label_length; if (!buffer.ReadUInt16(&label_length)) { LOG(LS_WARNING) << "Could not read OPEN message label length."; return false; } uint16 protocol_length; if (!buffer.ReadUInt16(&protocol_length)) { LOG(LS_WARNING) << "Could not read OPEN message protocol length."; return false; } if (!buffer.ReadString(label, (size_t) label_length)) { LOG(LS_WARNING) << "Could not read OPEN message label"; return false; } if (!buffer.ReadString(&config->protocol, protocol_length)) { LOG(LS_WARNING) << "Could not read OPEN message protocol."; return false; } config->ordered = true; switch (channel_type) { case DCOMCT_UNORDERED_RELIABLE: case DCOMCT_UNORDERED_PARTIAL_RTXS: case DCOMCT_UNORDERED_PARTIAL_TIME: config->ordered = false; } config->maxRetransmits = -1; config->maxRetransmitTime = -1; switch (channel_type) { case DCOMCT_ORDERED_PARTIAL_RTXS: case DCOMCT_UNORDERED_PARTIAL_RTXS: config->maxRetransmits = reliability_param; break; case DCOMCT_ORDERED_PARTIAL_TIME: case DCOMCT_UNORDERED_PARTIAL_TIME: config->maxRetransmitTime = reliability_param; break; } return true; } bool WriteDataChannelOpenMessage( const std::string& label, const webrtc::DataChannelInit& config, talk_base::Buffer* payload) { // Format defined at // http://tools.ietf.org/html/draft-ietf-rtcweb-data-protocol-00#section-6.1 uint8 channel_type = 0; uint32 reliability_param = 0; uint16 priority = 0; if (config.ordered) { if (config.maxRetransmits > -1) { channel_type = DCOMCT_ORDERED_PARTIAL_RTXS; reliability_param = config.maxRetransmits; } else if (config.maxRetransmitTime > -1) { channel_type = DCOMCT_ORDERED_PARTIAL_TIME; reliability_param = config.maxRetransmitTime; } else { channel_type = DCOMCT_ORDERED_RELIABLE; } } else { if (config.maxRetransmits > -1) { channel_type = DCOMCT_UNORDERED_PARTIAL_RTXS; reliability_param = config.maxRetransmits; } else if (config.maxRetransmitTime > -1) { channel_type = DCOMCT_UNORDERED_PARTIAL_TIME; reliability_param = config.maxRetransmitTime; } else { channel_type = DCOMCT_UNORDERED_RELIABLE; } } talk_base::ByteBuffer buffer( NULL, 20 + label.length() + config.protocol.length(), talk_base::ByteBuffer::ORDER_NETWORK); buffer.WriteUInt8(DATA_CHANNEL_OPEN_MESSAGE_TYPE); buffer.WriteUInt8(channel_type); buffer.WriteUInt16(priority); buffer.WriteUInt32(reliability_param); buffer.WriteUInt16(static_cast(label.length())); buffer.WriteUInt16(static_cast(config.protocol.length())); buffer.WriteString(label); buffer.WriteString(config.protocol); payload->SetData(buffer.Data(), buffer.Length()); return true; } } // namespace cricket