Index: extensions/browser/api/display_source/wifi_display/wifi_display_media_service_impl.cc |
diff --git a/extensions/browser/api/display_source/wifi_display/wifi_display_media_service_impl.cc b/extensions/browser/api/display_source/wifi_display/wifi_display_media_service_impl.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..203caa93b12cb97a0ebcd6bfdff9d26fed4cdede |
--- /dev/null |
+++ b/extensions/browser/api/display_source/wifi_display/wifi_display_media_service_impl.cc |
@@ -0,0 +1,134 @@ |
+// Copyright 2016 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 "extensions/browser/api/display_source/wifi_display/wifi_display_media_service_impl.h" |
+ |
+#include <string> |
+#include <vector> |
+ |
+#include "base/big_endian.h" |
+#include "content/public/browser/browser_thread.h" |
+#include "net/base/net_errors.h" |
+ |
+using content::BrowserThread; |
+ |
+namespace extensions { |
+ |
+class WiFiDisplayMediaServiceImpl::PacketIOBuffer : public net::IOBuffer { |
+ public: |
+ explicit PacketIOBuffer(mojo::Array<uint8_t> array); |
+ |
+ int size() const { return packet_data_.size(); } |
+ |
+ private: |
+ ~PacketIOBuffer() override; |
+ |
+ std::vector<uint8_t> packet_data_; |
+}; |
+ |
+WiFiDisplayMediaServiceImpl::PacketIOBuffer::PacketIOBuffer( |
+ mojo::Array<uint8_t> array) { |
+ array.Swap(&packet_data_); |
+ data_ = reinterpret_cast<char*>(packet_data_.data()); |
+} |
+ |
+WiFiDisplayMediaServiceImpl::PacketIOBuffer::~PacketIOBuffer() { |
+ data_ = nullptr; |
+} |
+ |
+// static |
+void WiFiDisplayMediaServiceImpl::Create( |
+ WiFiDisplayMediaServiceRequest request) { |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::IO); |
+ new WiFiDisplayMediaServiceImpl(std::move(request)); |
+} |
+ |
+// static |
+void WiFiDisplayMediaServiceImpl::BindToRequest( |
+ WiFiDisplayMediaServiceRequest request) { |
+ BrowserThread::PostTask(BrowserThread::IO, FROM_HERE, |
+ base::Bind(WiFiDisplayMediaServiceImpl::Create, |
+ base::Passed(std::move(request)))); |
+} |
+ |
+WiFiDisplayMediaServiceImpl::WiFiDisplayMediaServiceImpl( |
+ WiFiDisplayMediaServiceRequest request) |
+ : binding_(this, std::move(request)), |
+ last_send_code_(net::OK), |
+ weak_factory_(this) {} |
+ |
+WiFiDisplayMediaServiceImpl::~WiFiDisplayMediaServiceImpl() {} |
+ |
+void WiFiDisplayMediaServiceImpl::SetDesinationPoint( |
+ const mojo::String& ip_address, |
+ int32_t port, |
+ const SetDesinationPointCallback& callback) { |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::IO); |
+ net::IPAddress address; |
+ if (!address.AssignFromIPLiteral(std::string(ip_address))) { |
+ DVLOG(1) << "Failed to parse IP address from " << ip_address; |
+ callback.Run(false); |
+ return; |
+ } |
+ net::IPEndPoint end_point(address, static_cast<uint16_t>(port)); |
+ |
+ rtp_socket_.reset(new net::UDPSocket(net::DatagramSocket::DEFAULT_BIND, |
+ net::RandIntCallback(), nullptr, |
+ net::NetLog::Source())); |
+ if (rtp_socket_->Open(end_point.GetFamily()) != net::OK || |
+ rtp_socket_->Connect(end_point) != net::OK) { |
+ DVLOG(1) << "Could not connect to " << end_point.ToString(); |
+ callback.Run(false); |
+ rtp_socket_.reset(); |
+ return; |
+ } |
+ callback.Run(true); |
+} |
+ |
+void WiFiDisplayMediaServiceImpl::SendMediaPacket(mojo::Array<uint8_t> packet) { |
+ DCHECK(rtp_socket_); |
+ DCHECK_CURRENTLY_ON(content::BrowserThread::IO); |
+ |
+ if (packet.size() >> 15) { |
+ DVLOG(1) << "Packet size limit is exceeded, skipping."; |
+ return; |
+ } |
+ |
+ if (last_send_code_ == net::ERR_IO_PENDING) { |
+ VLOG(1) << "Cannot send because of pending IO, skipping"; |
+ return; |
+ } |
+ |
+ // Create, queue and send a write buffer. |
+ scoped_refptr<PacketIOBuffer> write_buffer = |
+ new PacketIOBuffer(std::move(packet)); |
+ write_buffers_.push(std::move(write_buffer)); |
+ |
+ Send(); |
+} |
+ |
+void WiFiDisplayMediaServiceImpl::Send() { |
+ DCHECK(!write_buffers_.empty()); |
+ last_send_code_ = rtp_socket_->Write( |
+ write_buffers_.front().get(), write_buffers_.front()->size(), |
+ base::Bind(&WiFiDisplayMediaServiceImpl::OnSent, |
+ weak_factory_.GetWeakPtr())); |
+ if (last_send_code_ != net::ERR_IO_PENDING) |
+ OnSent(last_send_code_); |
+} |
+ |
+void WiFiDisplayMediaServiceImpl::OnSent(int code) { |
+ last_send_code_ = code; |
+ if (code < 0) { |
+ VLOG(1) << "Unrepairable UDP socket error."; |
+ delete this; |
+ return; |
+ } |
+ DCHECK(!write_buffers_.empty()); |
+ write_buffers_.pop(); |
+ if (!write_buffers_.empty()) |
+ Send(); |
+} |
+ |
+} // namespace extensions |