2012-10-02 13:04:58 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=2 et sw=2 tw=80: */
|
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
|
|
|
|
// Original author: ekr@rtfm.com
|
|
|
|
|
|
|
|
// Some of this code is cut-and-pasted from nICEr. Copyright is:
|
|
|
|
|
|
|
|
/*
|
|
|
|
Copyright (c) 2007, Adobe Systems, Incorporated
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions are
|
|
|
|
met:
|
|
|
|
|
|
|
|
* Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
|
|
|
|
* 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.
|
|
|
|
|
|
|
|
* Neither the name of Adobe Systems, Network Resonance nor the names of its
|
|
|
|
contributors may be used to endorse or promote products derived from
|
|
|
|
this software without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT
|
|
|
|
OWNER OR 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 <string>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "nsError.h"
|
|
|
|
|
|
|
|
// nICEr includes
|
|
|
|
extern "C" {
|
|
|
|
#include "nr_api.h"
|
|
|
|
#include "registry.h"
|
|
|
|
#include "async_timer.h"
|
|
|
|
#include "ice_util.h"
|
|
|
|
#include "transport_addr.h"
|
|
|
|
#include "nr_crypto.h"
|
|
|
|
#include "nr_socket.h"
|
|
|
|
#include "nr_socket_local.h"
|
|
|
|
#include "stun_client_ctx.h"
|
|
|
|
#include "stun_server_ctx.h"
|
|
|
|
#include "ice_ctx.h"
|
|
|
|
#include "ice_candidate.h"
|
|
|
|
#include "ice_handler.h"
|
|
|
|
}
|
|
|
|
|
|
|
|
// Local includes
|
|
|
|
#include "logging.h"
|
|
|
|
#include "nricectx.h"
|
|
|
|
#include "nricemediastream.h"
|
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
|
2012-11-15 17:50:26 -08:00
|
|
|
MOZ_MTLOG_MODULE("mtransport")
|
2012-10-02 13:04:58 -07:00
|
|
|
|
|
|
|
// NrIceMediaStream
|
|
|
|
RefPtr<NrIceMediaStream>
|
|
|
|
NrIceMediaStream::Create(NrIceCtx *ctx,
|
|
|
|
const std::string& name,
|
|
|
|
int components) {
|
|
|
|
RefPtr<NrIceMediaStream> stream =
|
|
|
|
new NrIceMediaStream(ctx, name, components);
|
|
|
|
|
|
|
|
int r = nr_ice_add_media_stream(ctx->ctx(),
|
|
|
|
const_cast<char *>(name.c_str()),
|
|
|
|
components, &stream->stream_);
|
|
|
|
if (r) {
|
|
|
|
MOZ_MTLOG(PR_LOG_ERROR, "Couldn't create ICE media stream for '"
|
|
|
|
<< name << "'");
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
NrIceMediaStream::~NrIceMediaStream() {
|
|
|
|
// We do not need to destroy anything. All major resources
|
|
|
|
// are attached to the ice ctx.
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult NrIceMediaStream::ParseAttributes(std::vector<std::string>&
|
|
|
|
attributes) {
|
|
|
|
if (!stream_)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
std::vector<char *> attributes_in;
|
|
|
|
|
|
|
|
for (size_t i=0; i<attributes.size(); ++i) {
|
|
|
|
attributes_in.push_back(const_cast<char *>(attributes[i].c_str()));
|
|
|
|
}
|
2012-09-26 10:14:23 -07:00
|
|
|
|
2012-10-02 13:04:58 -07:00
|
|
|
// Still need to call nr_ice_ctx_parse_stream_attributes.
|
|
|
|
int r = nr_ice_peer_ctx_parse_stream_attributes(ctx_->peer(),
|
|
|
|
stream_,
|
|
|
|
attributes_in.size() ?
|
|
|
|
&attributes_in[0] : nullptr,
|
|
|
|
attributes_in.size());
|
|
|
|
if (r) {
|
|
|
|
MOZ_MTLOG(PR_LOG_ERROR, "Couldn't parse attributes for stream "
|
|
|
|
<< name_ << "'");
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse trickle ICE candidate
|
|
|
|
nsresult NrIceMediaStream::ParseTrickleCandidate(const std::string& candidate) {
|
|
|
|
int r;
|
|
|
|
|
2012-09-26 10:14:23 -07:00
|
|
|
MOZ_MTLOG(PR_LOG_DEBUG, "NrIceCtx(" << ctx_->name() << ")/STREAM(" <<
|
|
|
|
name() << ") : parsing trickle candidate " << candidate);
|
2012-10-02 13:04:58 -07:00
|
|
|
|
2013-01-17 09:18:06 -08:00
|
|
|
// Strip off Chrome's inappropriate leading a=.
|
|
|
|
// TODO(ekr@rtfm.com): remove once Chrome conforms to the specification
|
|
|
|
// or revisit if the specification changes.
|
|
|
|
char *candidate_ptr = const_cast<char *>(candidate.c_str());
|
|
|
|
std::string candidate_tmp; // Define out of if clause so it stays live.
|
|
|
|
if (!candidate.compare(0, 2, static_cast<const char *>("a="))) {
|
|
|
|
candidate_tmp = candidate.substr(2, std::string::npos);
|
|
|
|
candidate_ptr = const_cast<char *>(candidate_tmp.c_str());
|
|
|
|
}
|
2012-10-02 13:04:58 -07:00
|
|
|
r = nr_ice_peer_ctx_parse_trickle_candidate(ctx_->peer(),
|
2013-01-17 09:18:06 -08:00
|
|
|
stream_, candidate_ptr);
|
2012-10-02 13:04:58 -07:00
|
|
|
if (r) {
|
|
|
|
if (r == R_ALREADY) {
|
|
|
|
MOZ_MTLOG(PR_LOG_ERROR, "Trickle candidates are redundant for stream '"
|
|
|
|
<< name_ << "' because it is completed");
|
|
|
|
|
|
|
|
} else {
|
|
|
|
MOZ_MTLOG(PR_LOG_ERROR, "Couldn't parse trickle candidate for stream '"
|
|
|
|
<< name_ << "'");
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void NrIceMediaStream::EmitAllCandidates() {
|
|
|
|
char **attrs = 0;
|
|
|
|
int attrct;
|
|
|
|
int r;
|
|
|
|
r = nr_ice_media_stream_get_attributes(stream_,
|
|
|
|
&attrs, &attrct);
|
|
|
|
if (r) {
|
|
|
|
MOZ_MTLOG(PR_LOG_ERROR, "Couldn't get ICE candidates for '"
|
|
|
|
<< name_ << "'");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i=0; i<attrct; i++) {
|
|
|
|
SignalCandidate(this, attrs[i]);
|
|
|
|
RFREE(attrs[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
RFREE(attrs);
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult NrIceMediaStream::GetDefaultCandidate(int component,
|
|
|
|
std::string *addrp,
|
|
|
|
int *portp) {
|
|
|
|
nr_ice_candidate *cand;
|
|
|
|
int r;
|
|
|
|
|
|
|
|
r = nr_ice_media_stream_get_default_candidate(stream_,
|
|
|
|
component, &cand);
|
|
|
|
if (r) {
|
|
|
|
MOZ_MTLOG(PR_LOG_ERROR, "Couldn't get default ICE candidate for '"
|
|
|
|
<< name_ << "'");
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
char addr[64]; // Enough for IPv6 with colons.
|
|
|
|
r = nr_transport_addr_get_addrstring(&cand->addr,addr,sizeof(addr));
|
|
|
|
if (r)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
int port;
|
|
|
|
r=nr_transport_addr_get_port(&cand->addr,&port);
|
|
|
|
if (r)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
*addrp = addr;
|
|
|
|
*portp = port;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::string> NrIceMediaStream::GetCandidates() const {
|
|
|
|
char **attrs = 0;
|
|
|
|
int attrct;
|
|
|
|
int r;
|
|
|
|
std::vector<std::string> ret;
|
|
|
|
|
|
|
|
r = nr_ice_media_stream_get_attributes(stream_,
|
|
|
|
&attrs, &attrct);
|
|
|
|
if (r) {
|
|
|
|
MOZ_MTLOG(PR_LOG_ERROR, "Couldn't get ICE candidates for '"
|
|
|
|
<< name_ << "'");
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i=0; i<attrct; i++) {
|
|
|
|
ret.push_back(attrs[i]);
|
|
|
|
RFREE(attrs[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
RFREE(attrs);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsresult NrIceMediaStream::SendPacket(int component_id,
|
|
|
|
const unsigned char *data,
|
|
|
|
size_t len) {
|
|
|
|
if (!stream_)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
int r = nr_ice_media_stream_send(ctx_->peer(), stream_,
|
|
|
|
component_id,
|
|
|
|
const_cast<unsigned char *>(data), len);
|
|
|
|
if (r) {
|
|
|
|
MOZ_MTLOG(PR_LOG_ERROR, "Couldn't send media on '" << name_ << "'");
|
|
|
|
if (r == R_WOULDBLOCK) {
|
|
|
|
return NS_BASE_STREAM_WOULD_BLOCK;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_BASE_STREAM_OSERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void NrIceMediaStream::Ready() {
|
|
|
|
MOZ_MTLOG(PR_LOG_DEBUG, "Marking stream ready '" << name_ << "'");
|
|
|
|
state_ = ICE_OPEN;
|
|
|
|
SignalReady(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void NrIceMediaStream::Close() {
|
|
|
|
MOZ_MTLOG(PR_LOG_DEBUG, "Marking stream closed '" << name_ << "'");
|
|
|
|
state_ = ICE_CLOSED;
|
|
|
|
stream_ = nullptr;
|
|
|
|
}
|
|
|
|
} // close namespace
|