blob: 0a7307b446493e0178bd0dd08cf75caeec29645a [file] [log] [blame]
// Copyright 2015 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/host/desktop_capturer_proxy.h"
#include <stddef.h>
#include <utility>
#include "base/bind.h"
#include "base/location.h"
#include "base/macros.h"
#include "base/memory/ptr_util.h"
#include "base/single_thread_task_runner.h"
#include "base/threading/thread_task_runner_handle.h"
#include "remoting/proto/control.pb.h"
#include "third_party/webrtc/modules/desktop_capture/desktop_capture_options.h"
#include "third_party/webrtc/modules/desktop_capture/desktop_frame.h"
#include "third_party/webrtc/modules/desktop_capture/desktop_region.h"
#include "third_party/webrtc/modules/desktop_capture/screen_capturer.h"
#if defined(OS_CHROMEOS)
#include "remoting/host/chromeos/aura_desktop_capturer.h"
#endif
namespace remoting {
class DesktopCapturerProxy::Core : public webrtc::DesktopCapturer::Callback {
public:
explicit Core(base::WeakPtr<DesktopCapturerProxy> proxy);
~Core() override;
void CreateCapturer(const webrtc::DesktopCaptureOptions& options);
void Start();
void SetSharedMemoryFactory(
std::unique_ptr<webrtc::SharedMemoryFactory> shared_memory_factory);
void Capture(const webrtc::DesktopRegion& rect);
private:
// webrtc::DesktopCapturer::Callback implementation.
void OnCaptureCompleted(webrtc::DesktopFrame* frame) override;
base::ThreadChecker thread_checker_;
base::WeakPtr<DesktopCapturerProxy> proxy_;
scoped_refptr<base::SingleThreadTaskRunner> caller_task_runner_;
std::unique_ptr<webrtc::DesktopCapturer> capturer_;
DISALLOW_COPY_AND_ASSIGN(Core);
};
DesktopCapturerProxy::Core::Core(base::WeakPtr<DesktopCapturerProxy> proxy)
: proxy_(proxy), caller_task_runner_(base::ThreadTaskRunnerHandle::Get()) {
thread_checker_.DetachFromThread();
}
DesktopCapturerProxy::Core::~Core() {
DCHECK(thread_checker_.CalledOnValidThread());
}
void DesktopCapturerProxy::Core::CreateCapturer(
const webrtc::DesktopCaptureOptions& options) {
DCHECK(thread_checker_.CalledOnValidThread());
DCHECK(!capturer_);
#if defined(OS_CHROMEOS)
capturer_.reset(new AuraDesktopCapturer());
#else // !defined(OS_CHROMEOS)
capturer_.reset(webrtc::ScreenCapturer::Create(options));
#endif // !defined(OS_CHROMEOS)
if (!capturer_)
LOG(ERROR) << "Failed to initialize screen capturer.";
}
void DesktopCapturerProxy::Core::Start() {
DCHECK(thread_checker_.CalledOnValidThread());
if (capturer_)
capturer_->Start(this);
}
void DesktopCapturerProxy::Core::SetSharedMemoryFactory(
std::unique_ptr<webrtc::SharedMemoryFactory> shared_memory_factory) {
DCHECK(thread_checker_.CalledOnValidThread());
if (capturer_) {
capturer_->SetSharedMemoryFactory(std::move(shared_memory_factory));
}
}
void DesktopCapturerProxy::Core::Capture(const webrtc::DesktopRegion& rect) {
DCHECK(thread_checker_.CalledOnValidThread());
if (capturer_) {
capturer_->Capture(rect);
} else {
OnCaptureCompleted(nullptr);
}
}
void DesktopCapturerProxy::Core::OnCaptureCompleted(
webrtc::DesktopFrame* frame) {
DCHECK(thread_checker_.CalledOnValidThread());
caller_task_runner_->PostTask(
FROM_HERE, base::Bind(&DesktopCapturerProxy::OnFrameCaptured, proxy_,
base::Passed(base::WrapUnique(frame))));
}
DesktopCapturerProxy::DesktopCapturerProxy(
scoped_refptr<base::SingleThreadTaskRunner> capture_task_runner,
const webrtc::DesktopCaptureOptions& options)
: capture_task_runner_(capture_task_runner), weak_factory_(this) {
core_.reset(new Core(weak_factory_.GetWeakPtr()));
capture_task_runner_->PostTask(
FROM_HERE, base::Bind(&Core::CreateCapturer,
base::Unretained(core_.get()), options));
}
DesktopCapturerProxy::~DesktopCapturerProxy() {
capture_task_runner_->DeleteSoon(FROM_HERE, core_.release());
}
void DesktopCapturerProxy::Start(Callback* callback) {
DCHECK(thread_checker_.CalledOnValidThread());
callback_ = callback;
capture_task_runner_->PostTask(
FROM_HERE, base::Bind(&Core::Start, base::Unretained(core_.get())));
}
void DesktopCapturerProxy::SetSharedMemoryFactory(
std::unique_ptr<webrtc::SharedMemoryFactory> shared_memory_factory) {
DCHECK(thread_checker_.CalledOnValidThread());
capture_task_runner_->PostTask(
FROM_HERE,
base::Bind(
&Core::SetSharedMemoryFactory, base::Unretained(core_.get()),
base::Passed(base::WrapUnique(shared_memory_factory.release()))));
}
void DesktopCapturerProxy::Capture(const webrtc::DesktopRegion& rect) {
DCHECK(thread_checker_.CalledOnValidThread());
// Start() must be called before Capture().
DCHECK(callback_);
capture_task_runner_->PostTask(
FROM_HERE,
base::Bind(&Core::Capture, base::Unretained(core_.get()), rect));
}
void DesktopCapturerProxy::OnFrameCaptured(
std::unique_ptr<webrtc::DesktopFrame> frame) {
DCHECK(thread_checker_.CalledOnValidThread());
callback_->OnCaptureCompleted(frame.release());
}
} // namespace remoting