| 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
|
|
|