| OLD | NEW |
| (Empty) |
| 1 // Copyright 2013 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "media/cast/net/transport/transport.h" | |
| 6 | |
| 7 #include <string> | |
| 8 | |
| 9 #include "base/bind.h" | |
| 10 #include "base/logging.h" | |
| 11 #include "base/memory/ref_counted.h" | |
| 12 #include "base/memory/scoped_ptr.h" | |
| 13 #include "base/message_loop/message_loop.h" | |
| 14 #include "base/rand_util.h" | |
| 15 #include "net/base/io_buffer.h" | |
| 16 #include "net/base/rand_callback.h" | |
| 17 #include "net/base/test_completion_callback.h" | |
| 18 | |
| 19 namespace media { | |
| 20 namespace cast { | |
| 21 | |
| 22 const int kMaxPacketSize = 1500; | |
| 23 | |
| 24 class LocalUdpTransportData; | |
| 25 | |
| 26 void CreateUDPAddress(std::string ip_str, int port, net::IPEndPoint* address) { | |
| 27 net::IPAddressNumber ip_number; | |
| 28 bool rv = net::ParseIPLiteralToNumber(ip_str, &ip_number); | |
| 29 if (!rv) | |
| 30 return; | |
| 31 *address = net::IPEndPoint(ip_number, port); | |
| 32 } | |
| 33 | |
| 34 class LocalUdpTransportData | |
| 35 : public base::RefCountedThreadSafe<LocalUdpTransportData> { | |
| 36 public: | |
| 37 LocalUdpTransportData(net::UDPServerSocket* udp_socket, | |
| 38 scoped_refptr<base::TaskRunner> io_thread_proxy) | |
| 39 : udp_socket_(udp_socket), | |
| 40 buffer_(new net::IOBufferWithSize(kMaxPacketSize)), | |
| 41 io_thread_proxy_(io_thread_proxy) { | |
| 42 } | |
| 43 | |
| 44 void ListenTo(net::IPEndPoint bind_address) { | |
| 45 DCHECK(io_thread_proxy_->RunsTasksOnCurrentThread()); | |
| 46 | |
| 47 bind_address_ = bind_address; | |
| 48 io_thread_proxy_->PostTask(FROM_HERE, | |
| 49 base::Bind(&LocalUdpTransportData::RecvFromSocketLoop, this)); | |
| 50 } | |
| 51 | |
| 52 void DeletePacket(uint8* data) { | |
| 53 // Should be called from the receiver (not on the transport thread). | |
| 54 DCHECK(!(io_thread_proxy_->RunsTasksOnCurrentThread())); | |
| 55 delete [] data; | |
| 56 } | |
| 57 | |
| 58 void PacketReceived(int size) { | |
| 59 DCHECK(io_thread_proxy_->RunsTasksOnCurrentThread()); | |
| 60 // Got a packet with length result. | |
| 61 uint8* data = new uint8[size]; | |
| 62 memcpy(data, buffer_->data(), size); | |
| 63 packet_receiver_->ReceivedPacket(data, size, | |
| 64 base::Bind(&LocalUdpTransportData::DeletePacket, this, data)); | |
| 65 RecvFromSocketLoop(); | |
| 66 | |
| 67 } | |
| 68 | |
| 69 void RecvFromSocketLoop() { | |
| 70 DCHECK(io_thread_proxy_->RunsTasksOnCurrentThread()); | |
| 71 // Callback should always trigger with a packet. | |
| 72 int res = udp_socket_->RecvFrom(buffer_.get(), kMaxPacketSize, | |
| 73 &bind_address_, base::Bind(&LocalUdpTransportData::PacketReceived, | |
| 74 this)); | |
| 75 DCHECK(res >= net::ERR_IO_PENDING); | |
| 76 if (res > 0) { | |
| 77 PacketReceived(res); | |
| 78 } | |
| 79 } | |
| 80 | |
| 81 void set_packet_receiver(PacketReceiver* packet_receiver) { | |
| 82 packet_receiver_ = packet_receiver; | |
| 83 } | |
| 84 | |
| 85 void Close() { | |
| 86 udp_socket_->Close(); | |
| 87 } | |
| 88 | |
| 89 protected: | |
| 90 virtual ~LocalUdpTransportData() {} | |
| 91 | |
| 92 private: | |
| 93 friend class base::RefCountedThreadSafe<LocalUdpTransportData>; | |
| 94 | |
| 95 net::UDPServerSocket* udp_socket_; | |
| 96 net::IPEndPoint bind_address_; | |
| 97 PacketReceiver* packet_receiver_; | |
| 98 scoped_refptr<net::IOBufferWithSize> buffer_; | |
| 99 scoped_refptr<base::TaskRunner> io_thread_proxy_; | |
| 100 | |
| 101 DISALLOW_COPY_AND_ASSIGN(LocalUdpTransportData); | |
| 102 }; | |
| 103 | |
| 104 class LocalPacketSender : public PacketSender, | |
| 105 public base::RefCountedThreadSafe<LocalPacketSender> { | |
| 106 public: | |
| 107 LocalPacketSender(net::UDPServerSocket* udp_socket, | |
| 108 scoped_refptr<base::TaskRunner> io_thread_proxy) | |
| 109 : udp_socket_(udp_socket), | |
| 110 send_address_(), | |
| 111 io_thread_proxy_(io_thread_proxy) {} | |
| 112 | |
| 113 virtual bool SendPacket(const Packet& packet) OVERRIDE { | |
| 114 io_thread_proxy_->PostTask(FROM_HERE, | |
| 115 base::Bind(&LocalPacketSender::SendPacketToNetwork, this, packet)); | |
| 116 return true; | |
| 117 } | |
| 118 | |
| 119 virtual void SendPacketToNetwork(const Packet& packet) { | |
| 120 DCHECK(io_thread_proxy_->RunsTasksOnCurrentThread()); | |
| 121 const uint8* data = packet.data(); | |
| 122 net::TestCompletionCallback callback; | |
| 123 scoped_refptr<net::WrappedIOBuffer> buffer( | |
| 124 new net::WrappedIOBuffer(reinterpret_cast<const char*>(data))); | |
| 125 udp_socket_->SendTo(buffer.get(), static_cast<int>(packet.size()), | |
| 126 send_address_, callback.callback()); | |
| 127 } | |
| 128 | |
| 129 virtual bool SendPackets(const PacketList& packets) OVERRIDE { | |
| 130 bool out_val = true; | |
| 131 for (size_t i = 0; i < packets.size(); ++i) { | |
| 132 const Packet& packet = packets[i]; | |
| 133 out_val |= SendPacket(packet); | |
| 134 } | |
| 135 return out_val; | |
| 136 } | |
| 137 | |
| 138 void SetSendAddress(const net::IPEndPoint& send_address) { | |
| 139 send_address_ = send_address; | |
| 140 } | |
| 141 | |
| 142 protected: | |
| 143 virtual ~LocalPacketSender() {} | |
| 144 | |
| 145 private: | |
| 146 friend class base::RefCountedThreadSafe<LocalPacketSender>; | |
| 147 | |
| 148 net::UDPServerSocket* udp_socket_; // Not owned by this class. | |
| 149 net::IPEndPoint send_address_; | |
| 150 scoped_refptr<base::TaskRunner> io_thread_proxy_; | |
| 151 }; | |
| 152 | |
| 153 Transport::Transport( | |
| 154 scoped_refptr<base::TaskRunner> io_thread_proxy) | |
| 155 : udp_socket_(new net::UDPServerSocket(NULL, net::NetLog::Source())), | |
| 156 local_udp_transport_data_(new LocalUdpTransportData(udp_socket_.get(), | |
| 157 io_thread_proxy)), | |
| 158 packet_sender_(new LocalPacketSender(udp_socket_.get(), io_thread_proxy)), | |
| 159 io_thread_proxy_(io_thread_proxy) {} | |
| 160 | |
| 161 Transport::~Transport() {} | |
| 162 | |
| 163 PacketSender* Transport::packet_sender() { | |
| 164 return static_cast<PacketSender*>(packet_sender_.get()); | |
| 165 } | |
| 166 | |
| 167 void Transport::StopReceiving() { | |
| 168 local_udp_transport_data_->Close(); | |
| 169 } | |
| 170 | |
| 171 void Transport::SetLocalReceiver(PacketReceiver* packet_receiver, | |
| 172 std::string ip_address, | |
| 173 std::string local_ip_address, | |
| 174 int port) { | |
| 175 DCHECK(io_thread_proxy_->RunsTasksOnCurrentThread()); | |
| 176 net::IPEndPoint bind_address, local_bind_address; | |
| 177 CreateUDPAddress(ip_address, port, &bind_address); | |
| 178 CreateUDPAddress(local_ip_address, port, &local_bind_address); | |
| 179 local_udp_transport_data_->set_packet_receiver(packet_receiver); | |
| 180 udp_socket_->AllowAddressReuse(); | |
| 181 udp_socket_->SetMulticastLoopbackMode(true); | |
| 182 udp_socket_->Listen(local_bind_address); | |
| 183 | |
| 184 // Start listening once receiver has been set. | |
| 185 local_udp_transport_data_->ListenTo(bind_address); | |
| 186 } | |
| 187 | |
| 188 void Transport::SetSendDestination(std::string ip_address, int port) { | |
| 189 net::IPEndPoint send_address; | |
| 190 CreateUDPAddress(ip_address, port, &send_address); | |
| 191 packet_sender_->SetSendAddress(send_address); | |
| 192 } | |
| 193 | |
| 194 } // namespace cast | |
| 195 } // namespace media | |
| OLD | NEW |