зеркало из https://github.com/mozilla/gecko-dev.git
318 строки
11 KiB
C++
318 строки
11 KiB
C++
/* -*- 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 source code here 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.
|
|
*/
|
|
|
|
// Implementation of nICEr/nr_socket that is tied to the Gecko
|
|
// SocketTransportService.
|
|
|
|
#ifndef nr_socket_prsock__
|
|
#define nr_socket_prsock__
|
|
|
|
#include <memory>
|
|
#include <queue>
|
|
|
|
#include "nspr.h"
|
|
#include "prio.h"
|
|
|
|
#include "nsCOMPtr.h"
|
|
#include "nsASocketHandler.h"
|
|
#include "nsXPCOM.h"
|
|
#include "nsIEventTarget.h"
|
|
#include "nsIUDPSocketChild.h"
|
|
#include "nsProxyRelease.h"
|
|
#include "nsThreadUtils.h"
|
|
|
|
#include "mediapacket.h"
|
|
#include "m_cpp_utils.h"
|
|
#include "mozilla/ReentrantMonitor.h"
|
|
#include "mozilla/RefPtr.h"
|
|
#include "mozilla/TimeStamp.h"
|
|
#include "mozilla/ClearOnShutdown.h"
|
|
|
|
// nICEr includes
|
|
extern "C" {
|
|
#include "transport_addr.h"
|
|
#include "async_wait.h"
|
|
}
|
|
|
|
// Stub declaration for nICEr type
|
|
typedef struct nr_socket_vtbl_ nr_socket_vtbl;
|
|
typedef struct nr_socket_ nr_socket;
|
|
|
|
#if defined(MOZILLA_INTERNAL_API)
|
|
namespace mozilla {
|
|
class NrSocketProxyConfig;
|
|
} // namespace mozilla
|
|
#endif
|
|
|
|
namespace mozilla {
|
|
|
|
namespace net {
|
|
union NetAddr;
|
|
}
|
|
|
|
namespace dom {
|
|
class UDPSocketChild;
|
|
} // namespace dom
|
|
|
|
class NrSocketBase {
|
|
public:
|
|
NrSocketBase() : connect_invoked_(false), poll_flags_(0) {
|
|
memset(cbs_, 0, sizeof(cbs_));
|
|
memset(cb_args_, 0, sizeof(cb_args_));
|
|
memset(&my_addr_, 0, sizeof(my_addr_));
|
|
}
|
|
virtual ~NrSocketBase() = default;
|
|
|
|
// Factory method; will create either an NrSocket, NrUdpSocketIpc, or
|
|
// NrTcpSocketIpc as appropriate.
|
|
static int CreateSocket(nr_transport_addr* addr, RefPtr<NrSocketBase>* sock,
|
|
const std::shared_ptr<NrSocketProxyConfig>& config);
|
|
static bool IsForbiddenAddress(nr_transport_addr* addr);
|
|
|
|
// the nr_socket APIs
|
|
virtual int create(nr_transport_addr* addr) = 0;
|
|
virtual int sendto(const void* msg, size_t len, int flags,
|
|
nr_transport_addr* to) = 0;
|
|
virtual int recvfrom(void* buf, size_t maxlen, size_t* len, int flags,
|
|
nr_transport_addr* from) = 0;
|
|
virtual int getaddr(nr_transport_addr* addrp) = 0;
|
|
virtual void close() = 0;
|
|
virtual int connect(nr_transport_addr* addr) = 0;
|
|
virtual int write(const void* msg, size_t len, size_t* written) = 0;
|
|
virtual int read(void* buf, size_t maxlen, size_t* len) = 0;
|
|
virtual int listen(int backlog) = 0;
|
|
virtual int accept(nr_transport_addr* addrp, nr_socket** sockp) = 0;
|
|
|
|
// Implementations of the async_event APIs
|
|
virtual int async_wait(int how, NR_async_cb cb, void* cb_arg, char* function,
|
|
int line);
|
|
virtual int cancel(int how);
|
|
|
|
// nsISupport reference counted interface
|
|
NS_INLINE_DECL_PURE_VIRTUAL_REFCOUNTING
|
|
|
|
uint32_t poll_flags() { return poll_flags_; }
|
|
|
|
virtual nr_socket_vtbl* vtbl(); // To access in test classes.
|
|
|
|
static TimeStamp short_term_violation_time();
|
|
static TimeStamp long_term_violation_time();
|
|
const nr_transport_addr& my_addr() const { return my_addr_; }
|
|
|
|
void fire_callback(int how);
|
|
|
|
protected:
|
|
bool connect_invoked_;
|
|
nr_transport_addr my_addr_;
|
|
|
|
private:
|
|
NR_async_cb cbs_[NR_ASYNC_WAIT_WRITE + 1];
|
|
void* cb_args_[NR_ASYNC_WAIT_WRITE + 1];
|
|
uint32_t poll_flags_;
|
|
};
|
|
|
|
class NrSocket : public NrSocketBase, public nsASocketHandler {
|
|
public:
|
|
NrSocket() : fd_(nullptr) {}
|
|
|
|
// Implement nsASocket
|
|
virtual void OnSocketReady(PRFileDesc* fd, int16_t outflags) override;
|
|
virtual void OnSocketDetached(PRFileDesc* fd) override;
|
|
virtual void IsLocal(bool* aIsLocal) override;
|
|
virtual uint64_t ByteCountSent() override { return 0; }
|
|
virtual uint64_t ByteCountReceived() override { return 0; }
|
|
|
|
// nsISupports methods
|
|
NS_DECL_THREADSAFE_ISUPPORTS
|
|
|
|
// Implementations of the async_event APIs
|
|
virtual int async_wait(int how, NR_async_cb cb, void* cb_arg, char* function,
|
|
int line) override;
|
|
virtual int cancel(int how) override;
|
|
|
|
// Implementations of the nr_socket APIs
|
|
virtual int create(nr_transport_addr* addr)
|
|
override; // (really init, but it's called create)
|
|
virtual int sendto(const void* msg, size_t len, int flags,
|
|
nr_transport_addr* to) override;
|
|
virtual int recvfrom(void* buf, size_t maxlen, size_t* len, int flags,
|
|
nr_transport_addr* from) override;
|
|
virtual int getaddr(nr_transport_addr* addrp) override;
|
|
virtual void close() override;
|
|
virtual int connect(nr_transport_addr* addr) override;
|
|
virtual int write(const void* msg, size_t len, size_t* written) override;
|
|
virtual int read(void* buf, size_t maxlen, size_t* len) override;
|
|
virtual int listen(int backlog) override;
|
|
virtual int accept(nr_transport_addr* addrp, nr_socket** sockp) override;
|
|
|
|
protected:
|
|
virtual ~NrSocket() {
|
|
if (fd_) PR_Close(fd_);
|
|
}
|
|
|
|
DISALLOW_COPY_ASSIGN(NrSocket);
|
|
|
|
PRFileDesc* fd_;
|
|
nsCOMPtr<nsIEventTarget> ststhread_;
|
|
};
|
|
|
|
struct nr_udp_message {
|
|
nr_udp_message(const PRNetAddr& from, UniquePtr<MediaPacket>&& data)
|
|
: from(from), data(std::move(data)) {}
|
|
|
|
NS_INLINE_DECL_THREADSAFE_REFCOUNTING(nr_udp_message);
|
|
|
|
PRNetAddr from;
|
|
UniquePtr<MediaPacket> data;
|
|
|
|
private:
|
|
~nr_udp_message() = default;
|
|
DISALLOW_COPY_ASSIGN(nr_udp_message);
|
|
};
|
|
|
|
class NrSocketIpc : public NrSocketBase {
|
|
public:
|
|
enum NrSocketIpcState {
|
|
NR_INIT,
|
|
NR_CONNECTING,
|
|
NR_CONNECTED,
|
|
NR_CLOSING,
|
|
NR_CLOSED,
|
|
};
|
|
|
|
NrSocketIpc(nsIEventTarget* aThread);
|
|
|
|
protected:
|
|
nsCOMPtr<nsIEventTarget> sts_thread_;
|
|
// Note: for UDP PBackground, this is a thread held by SingletonThreadHolder.
|
|
// For TCP PNecko, this is MainThread (and TCPSocket requires MainThread
|
|
// currently)
|
|
const nsCOMPtr<nsIEventTarget> io_thread_;
|
|
virtual ~NrSocketIpc() = default;
|
|
|
|
private:
|
|
DISALLOW_COPY_ASSIGN(NrSocketIpc);
|
|
};
|
|
|
|
class NrUdpSocketIpc : public NrSocketIpc {
|
|
public:
|
|
NS_INLINE_DECL_THREADSAFE_REFCOUNTING(NrUdpSocketIpc, override)
|
|
|
|
NS_IMETHODIMP CallListenerError(const nsACString& message,
|
|
const nsACString& filename,
|
|
uint32_t line_number);
|
|
NS_IMETHODIMP CallListenerReceivedData(const nsACString& host, uint16_t port,
|
|
const nsTArray<uint8_t>& data);
|
|
NS_IMETHODIMP CallListenerOpened();
|
|
NS_IMETHODIMP CallListenerConnected();
|
|
NS_IMETHODIMP CallListenerClosed();
|
|
|
|
NrUdpSocketIpc();
|
|
|
|
// Implementations of the NrSocketBase APIs
|
|
virtual int create(nr_transport_addr* addr) override;
|
|
virtual int sendto(const void* msg, size_t len, int flags,
|
|
nr_transport_addr* to) override;
|
|
virtual int recvfrom(void* buf, size_t maxlen, size_t* len, int flags,
|
|
nr_transport_addr* from) override;
|
|
virtual int getaddr(nr_transport_addr* addrp) override;
|
|
virtual void close() override;
|
|
virtual int connect(nr_transport_addr* addr) override;
|
|
virtual int write(const void* msg, size_t len, size_t* written) override;
|
|
virtual int read(void* buf, size_t maxlen, size_t* len) override;
|
|
virtual int listen(int backlog) override;
|
|
virtual int accept(nr_transport_addr* addrp, nr_socket** sockp) override;
|
|
|
|
private:
|
|
virtual ~NrUdpSocketIpc();
|
|
|
|
DISALLOW_COPY_ASSIGN(NrUdpSocketIpc);
|
|
|
|
nsresult SetAddress(); // Set the local address from parent info.
|
|
|
|
// Main or private thread executors of the NrSocketBase APIs
|
|
void create_i(const nsACString& host, const uint16_t port);
|
|
void connect_i(const nsACString& host, const uint16_t port);
|
|
void sendto_i(const net::NetAddr& addr, UniquePtr<MediaPacket> buf);
|
|
void close_i();
|
|
#if defined(MOZILLA_INTERNAL_API) && !defined(MOZILLA_XPCOMRT_API)
|
|
static void destroy_i(dom::UDPSocketChild* aChild,
|
|
const nsCOMPtr<nsIEventTarget>& aStsThread);
|
|
#endif
|
|
// STS thread executor
|
|
void recv_callback_s(RefPtr<nr_udp_message> msg);
|
|
|
|
ReentrantMonitor monitor_; // protects err_and state_
|
|
bool err_;
|
|
NrSocketIpcState state_;
|
|
|
|
std::queue<RefPtr<nr_udp_message>> received_msgs_;
|
|
|
|
// only accessed from the io_thread
|
|
RefPtr<dom::UDPSocketChild> socket_child_;
|
|
};
|
|
|
|
// The socket child holds onto one of these, which just passes callbacks
|
|
// through and makes sure the ref to the NrSocketIpc is released on STS.
|
|
class NrUdpSocketIpcProxy : public nsIUDPSocketInternal {
|
|
public:
|
|
NS_DECL_THREADSAFE_ISUPPORTS
|
|
NS_DECL_NSIUDPSOCKETINTERNAL
|
|
|
|
nsresult Init(const RefPtr<NrUdpSocketIpc>& socket);
|
|
|
|
private:
|
|
virtual ~NrUdpSocketIpcProxy();
|
|
|
|
RefPtr<NrUdpSocketIpc> socket_;
|
|
nsCOMPtr<nsIEventTarget> sts_thread_;
|
|
};
|
|
|
|
int nr_netaddr_to_transport_addr(const net::NetAddr* netaddr,
|
|
nr_transport_addr* addr, int protocol);
|
|
int nr_praddr_to_transport_addr(const PRNetAddr* praddr,
|
|
nr_transport_addr* addr, int protocol,
|
|
int keep);
|
|
int nr_transport_addr_get_addrstring_and_port(nr_transport_addr* addr,
|
|
nsACString* host, int32_t* port);
|
|
} // namespace mozilla
|
|
#endif
|