// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "remoting/jingle_glue/chromium_socket_factory.h" #include "base/bind.h" #include "base/logging.h" #include "base/memory/scoped_ptr.h" #include "jingle/glue/utils.h" #include "net/base/io_buffer.h" #include "net/base/ip_endpoint.h" #include "net/base/net_errors.h" #include "net/udp/udp_server_socket.h" #include "third_party/libjingle/source/talk/base/asyncpacketsocket.h" namespace remoting { namespace { // Size of the buffer to allocate for RecvFrom(). const int kReceiveBufferSize = 65536; // Maximum amount of data in the send buffers. This is necessary to // prevent out-of-memory crashes if the caller sends data faster than // Pepper's UDP API can handle it. This maximum should never be // reached under normal conditions. const int kMaxSendBufferSize = 256 * 1024; // Defines set of transient errors. These errors are ignored when we get them // from sendto() calls. bool IsTransientError(int error) { return error == net::ERR_ADDRESS_UNREACHABLE || error == net::ERR_ADDRESS_INVALID; } class UdpPacketSocket : public talk_base::AsyncPacketSocket { public: UdpPacketSocket(); virtual ~UdpPacketSocket(); bool Init(const talk_base::SocketAddress& local_address, int min_port, int max_port); // talk_base::AsyncPacketSocket interface. virtual talk_base::SocketAddress GetLocalAddress() const OVERRIDE; virtual talk_base::SocketAddress GetRemoteAddress() const OVERRIDE; virtual int Send(const void* data, size_t data_size) OVERRIDE; virtual int SendTo(const void* data, size_t data_size, const talk_base::SocketAddress& address) OVERRIDE; virtual int Close() OVERRIDE; virtual State GetState() const OVERRIDE; virtual int GetOption(talk_base::Socket::Option option, int* value) OVERRIDE; virtual int SetOption(talk_base::Socket::Option option, int value) OVERRIDE; virtual int GetError() const OVERRIDE; virtual void SetError(int error) OVERRIDE; private: struct PendingPacket { PendingPacket(const void* buffer, int buffer_size, const net::IPEndPoint& address); scoped_refptr data; net::IPEndPoint address; }; void OnBindCompleted(int error); void DoSend(); void OnSendCompleted(int result); void DoRead(); void OnReadCompleted(int result); void HandleReadResult(int result); scoped_ptr socket_; State state_; int error_; talk_base::SocketAddress local_address_; // Receive buffer and address are populated by asynchronous reads. scoped_refptr receive_buffer_; net::IPEndPoint receive_address_; bool send_pending_; std::list send_queue_; int send_queue_size_; DISALLOW_COPY_AND_ASSIGN(UdpPacketSocket); }; UdpPacketSocket::PendingPacket::PendingPacket( const void* buffer, int buffer_size, const net::IPEndPoint& address) : data(new net::IOBufferWithSize(buffer_size)), address(address) { memcpy(data->data(), buffer, buffer_size); } UdpPacketSocket::UdpPacketSocket() : state_(STATE_CLOSED), error_(0), send_pending_(false), send_queue_size_(0) { } UdpPacketSocket::~UdpPacketSocket() { Close(); } bool UdpPacketSocket::Init(const talk_base::SocketAddress& local_address, int min_port, int max_port) { net::IPEndPoint local_endpoint; if (!jingle_glue::SocketAddressToIPEndPoint( local_address, &local_endpoint)) { return false; } for (int port = min_port; port <= max_port; ++port) { socket_.reset(new net::UDPServerSocket(NULL, net::NetLog::Source())); int result = socket_->Listen( net::IPEndPoint(local_endpoint.address(), port)); if (result == net::OK) { break; } else { socket_.reset(); } } if (!socket_.get()) { // Failed to bind the socket. return false; } if (socket_->GetLocalAddress(&local_endpoint) != net::OK || !jingle_glue::IPEndPointToSocketAddress(local_endpoint, &local_address_)) { return false; } state_ = STATE_BOUND; DoRead(); return true; } talk_base::SocketAddress UdpPacketSocket::GetLocalAddress() const { DCHECK_EQ(state_, STATE_BOUND); return local_address_; } talk_base::SocketAddress UdpPacketSocket::GetRemoteAddress() const { // UDP sockets are not connected - this method should never be called. NOTREACHED(); return talk_base::SocketAddress(); } int UdpPacketSocket::Send(const void* data, size_t data_size) { // UDP sockets are not connected - this method should never be called. NOTREACHED(); return EWOULDBLOCK; } int UdpPacketSocket::SendTo(const void* data, size_t data_size, const talk_base::SocketAddress& address) { if (state_ != STATE_BOUND) { NOTREACHED(); return EINVAL; } if (error_ != 0) { return error_; } net::IPEndPoint endpoint; if (!jingle_glue::SocketAddressToIPEndPoint(address, &endpoint)) { return EINVAL; } if (send_queue_size_ >= kMaxSendBufferSize) { return EWOULDBLOCK; } send_queue_.push_back(PendingPacket(data, data_size, endpoint)); send_queue_size_ += data_size; DoSend(); return data_size; } int UdpPacketSocket::Close() { state_ = STATE_CLOSED; socket_.reset(); return 0; } talk_base::AsyncPacketSocket::State UdpPacketSocket::GetState() const { return state_; } int UdpPacketSocket::GetOption(talk_base::Socket::Option option, int* value) { // This method is never called by libjingle. NOTIMPLEMENTED(); return -1; } int UdpPacketSocket::SetOption(talk_base::Socket::Option option, int value) { if (state_ != STATE_BOUND) { NOTREACHED(); return EINVAL; } switch (option) { case talk_base::Socket::OPT_DONTFRAGMENT: NOTIMPLEMENTED(); return -1; case talk_base::Socket::OPT_RCVBUF: { bool success = socket_->SetReceiveBufferSize(value); return success ? 0 : -1; } case talk_base::Socket::OPT_SNDBUF: { bool success = socket_->SetSendBufferSize(value); return success ? 0 : -1; } case talk_base::Socket::OPT_NODELAY: // OPT_NODELAY is only for TCP sockets. NOTREACHED(); return -1; case talk_base::Socket::OPT_IPV6_V6ONLY: NOTIMPLEMENTED(); return -1; } NOTREACHED(); return -1; } int UdpPacketSocket::GetError() const { return error_; } void UdpPacketSocket::SetError(int error) { error_ = error; } void UdpPacketSocket::DoSend() { if (send_pending_ || send_queue_.empty()) return; PendingPacket& packet = send_queue_.front(); int result = socket_->SendTo( packet.data, packet.data->size(), packet.address, base::Bind(&UdpPacketSocket::OnSendCompleted, base::Unretained(this))); if (result == net::ERR_IO_PENDING) { send_pending_ = true; } else { OnSendCompleted(result); } } void UdpPacketSocket::OnSendCompleted(int result) { send_pending_ = false; if (result < 0) { if (!IsTransientError(result)) { LOG(ERROR) << "Send failed on a UDP socket: " << result; error_ = EINVAL; return; } } // Don't need to worry about partial sends because this is a datagram // socket. send_queue_size_ -= send_queue_.front().data->size(); send_queue_.pop_front(); DoSend(); } void UdpPacketSocket::DoRead() { int result = 0; while (result >= 0) { receive_buffer_ = new net::IOBuffer(kReceiveBufferSize); result = socket_->RecvFrom( receive_buffer_, kReceiveBufferSize, &receive_address_, base::Bind(&UdpPacketSocket::OnReadCompleted, base::Unretained(this))); HandleReadResult(result); } } void UdpPacketSocket::OnReadCompleted(int result) { HandleReadResult(result); if (result >= 0) { DoRead(); } } void UdpPacketSocket::HandleReadResult(int result) { if (result == net::ERR_IO_PENDING) { return; } if (result > 0) { talk_base::SocketAddress address; if (!jingle_glue::IPEndPointToSocketAddress(receive_address_, &address)) { NOTREACHED(); LOG(ERROR) << "Failed to convert address received from RecvFrom()."; return; } SignalReadPacket(this, receive_buffer_->data(), result, address); } else { LOG(ERROR) << "Received error when reading from UDP socket: " << result; } } } // namespace ChromiumPacketSocketFactory::ChromiumPacketSocketFactory() { } ChromiumPacketSocketFactory::~ChromiumPacketSocketFactory() { } talk_base::AsyncPacketSocket* ChromiumPacketSocketFactory::CreateUdpSocket( const talk_base::SocketAddress& local_address, int min_port, int max_port) { scoped_ptr result(new UdpPacketSocket()); if (!result->Init(local_address, min_port, max_port)) return NULL; return result.release(); } talk_base::AsyncPacketSocket* ChromiumPacketSocketFactory::CreateServerTcpSocket( const talk_base::SocketAddress& local_address, int min_port, int max_port, bool ssl) { // We don't use TCP sockets for remoting connections. NOTREACHED(); return NULL; } talk_base::AsyncPacketSocket* ChromiumPacketSocketFactory::CreateClientTcpSocket( const talk_base::SocketAddress& local_address, const talk_base::SocketAddress& remote_address, const talk_base::ProxyInfo& proxy_info, const std::string& user_agent, bool ssl) { // We don't use TCP sockets for remoting connections. NOTREACHED(); return NULL; } } // namespace remoting