Add DesktopCaptureOptions class.
The new class is used to pass configuration parameters to screen/window
capturers. It also allows to share X Window connection between multiple
objects.
R=wez@chromium.org
Review URL: https://webrtc-codereview.appspot.com/2374004
git-svn-id: http://webrtc.googlecode.com/svn/trunk@4952 4adac7df-926f-26a2-2b94-8c16560cd09d
diff --git a/webrtc/modules/desktop_capture/desktop_capture.gypi b/webrtc/modules/desktop_capture/desktop_capture.gypi
index 0bc3839..50084c9 100644
--- a/webrtc/modules/desktop_capture/desktop_capture.gypi
+++ b/webrtc/modules/desktop_capture/desktop_capture.gypi
@@ -22,6 +22,9 @@
"desktop_frame_win.h",
"desktop_geometry.cc",
"desktop_geometry.h",
+ "desktop_capture_options.h",
+ "desktop_capture_options.cc",
+ "desktop_capturer.h",
"desktop_region.cc",
"desktop_region.h",
"differ.cc",
@@ -35,6 +38,7 @@
"mouse_cursor_shape.h",
"screen_capture_frame_queue.cc",
"screen_capture_frame_queue.h",
+ "screen_capturer.cc",
"screen_capturer.h",
"screen_capturer_helper.cc",
"screen_capturer_helper.h",
@@ -52,10 +56,13 @@
"win/scoped_gdi_object.h",
"win/scoped_thread_desktop.cc",
"win/scoped_thread_desktop.h",
+ "window_capturer.cc",
"window_capturer.h",
"window_capturer_mac.cc",
"window_capturer_win.cc",
"window_capturer_x11.cc",
+ "x11/shared_x_display.h",
+ "x11/shared_x_display.cc",
"x11/x_error_trap.cc",
"x11/x_error_trap.h",
"x11/x_server_pixel_buffer.cc",
diff --git a/webrtc/modules/desktop_capture/desktop_capture_options.cc b/webrtc/modules/desktop_capture/desktop_capture_options.cc
new file mode 100644
index 0000000..a4fa025
--- /dev/null
+++ b/webrtc/modules/desktop_capture/desktop_capture_options.cc
@@ -0,0 +1,35 @@
+/*
+ * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#include "webrtc/modules/desktop_capture/desktop_capture_options.h"
+
+namespace webrtc {
+
+DesktopCaptureOptions::DesktopCaptureOptions()
+ : use_update_notifications_(true),
+ disable_effects_(true) {
+#if defined(USE_X11)
+ // XDamage is often broken, so don't use it by default.
+ use_update_notifications_ = false;
+#endif
+}
+
+DesktopCaptureOptions::~DesktopCaptureOptions() {}
+
+// static
+DesktopCaptureOptions DesktopCaptureOptions::CreateDefault() {
+ DesktopCaptureOptions result;
+#if defined(USE_X11)
+ result.set_x_display(SharedXDisplay::CreateDefault());
+#endif
+ return result;
+}
+
+} // namespace webrtc
diff --git a/webrtc/modules/desktop_capture/desktop_capture_options.h b/webrtc/modules/desktop_capture/desktop_capture_options.h
new file mode 100644
index 0000000..f0c76b1
--- /dev/null
+++ b/webrtc/modules/desktop_capture/desktop_capture_options.h
@@ -0,0 +1,65 @@
+/*
+ * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+#ifndef WEBRTC_MODULES_DESKTOP_CAPTURE_DESKTOP_CAPTURE_OPTIONS_H_
+#define WEBRTC_MODULES_DESKTOP_CAPTURE_DESKTOP_CAPTURE_OPTIONS_H_
+
+#include "webrtc/system_wrappers/interface/constructor_magic.h"
+
+#if defined(USE_X11)
+#include "webrtc/modules/desktop_capture/x11/shared_x_display.h"
+#endif
+
+namespace webrtc {
+
+// An object that stores initialization parameters for screen and window
+// capturers.
+class DesktopCaptureOptions {
+ public:
+ // Creates an empty Options instance (e.g. without X display).
+ DesktopCaptureOptions();
+ ~DesktopCaptureOptions();
+
+ // Returns instance of DesktopCaptureOptions with default parameters. On Linux
+ // also initializes X window connection. x_display() will be set to null if
+ // X11 connection failed (e.g. DISPLAY isn't set).
+ static DesktopCaptureOptions CreateDefault();
+
+#if defined(USE_X11)
+ SharedXDisplay* x_display() const { return x_display_; }
+ void set_x_display(scoped_refptr<SharedXDisplay> x_display) {
+ x_display_ = x_display;
+ }
+#endif
+
+ // Flag indicating that the capturer should use screen change notifications.
+ // Enables/disables use of XDAMAGE in the X11 capturer.
+ bool use_update_notifications() const { return use_update_notifications_; }
+ void set_use_update_notifications(bool use_update_notifications) {
+ use_update_notifications_ = use_update_notifications;
+ }
+
+ // Flag indicating if desktop effects (e.g. Aero) should be disabled when the
+ // capturer is active. Currently used only on Windows.
+ bool disable_effects() const { return disable_effects_; }
+ void set_disable_effects(bool disable_effects) {
+ disable_effects_ = disable_effects;
+ }
+
+ private:
+#if defined(USE_X11)
+ scoped_refptr<SharedXDisplay> x_display_;
+#endif
+ bool use_update_notifications_;
+ bool disable_effects_;
+};
+
+} // namespace webrtc
+
+#endif // WEBRTC_MODULES_DESKTOP_CAPTURE_DESKTOP_CAPTURE_OPTIONS_H_
diff --git a/webrtc/modules/desktop_capture/screen_capturer.cc b/webrtc/modules/desktop_capture/screen_capturer.cc
new file mode 100644
index 0000000..97f69d3
--- /dev/null
+++ b/webrtc/modules/desktop_capture/screen_capturer.cc
@@ -0,0 +1,36 @@
+/*
+ * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#include "webrtc/modules/desktop_capture/screen_capturer.h"
+
+#include "webrtc/modules/desktop_capture/desktop_capture_options.h"
+
+namespace webrtc {
+
+ScreenCapturer* ScreenCapturer::Create() {
+ return Create(DesktopCaptureOptions::CreateDefault());
+}
+
+#if defined(WEBRTC_LINUX)
+ScreenCapturer* ScreenCapturer::CreateWithXDamage(
+ bool use_update_notifications) {
+ DesktopCaptureOptions options;
+ options.set_use_update_notifications(use_update_notifications);
+ return Create(options);
+}
+#elif defined(WEBRTC_WIN)
+ScreenCapturer* ScreenCapturer::CreateWithDisableAero(bool disable_effects) {
+ DesktopCaptureOptions options;
+ options.set_disable_effects(disable_effects);
+ return Create(options);
+}
+#endif
+
+} // namespace webrtc
diff --git a/webrtc/modules/desktop_capture/screen_capturer.h b/webrtc/modules/desktop_capture/screen_capturer.h
index 17101c5..9dd3a10 100644
--- a/webrtc/modules/desktop_capture/screen_capturer.h
+++ b/webrtc/modules/desktop_capture/screen_capturer.h
@@ -17,6 +17,7 @@
namespace webrtc {
+class DesktopCaptureOptions;
struct MouseCursorShape;
// Class used to capture video frames asynchronously.
@@ -57,6 +58,10 @@
virtual ~ScreenCapturer() {}
// Creates platform-specific capturer.
+ //
+ // TODO(sergeyu): Remove all Create() methods except the first one.
+ // crbug.com/172183
+ static ScreenCapturer* Create(const DesktopCaptureOptions& options);
static ScreenCapturer* Create();
#if defined(WEBRTC_LINUX)
diff --git a/webrtc/modules/desktop_capture/screen_capturer_mac.mm b/webrtc/modules/desktop_capture/screen_capturer_mac.mm
index 3165fd3..5cbffb0 100644
--- a/webrtc/modules/desktop_capture/screen_capturer_mac.mm
+++ b/webrtc/modules/desktop_capture/screen_capturer_mac.mm
@@ -21,6 +21,7 @@
#include <OpenGL/OpenGL.h>
#include <sys/utsname.h>
+#include "webrtc/modules/desktop_capture/desktop_capture_options.h"
#include "webrtc/modules/desktop_capture/desktop_frame.h"
#include "webrtc/modules/desktop_capture/desktop_geometry.h"
#include "webrtc/modules/desktop_capture/desktop_region.h"
@@ -895,7 +896,7 @@
} // namespace
// static
-ScreenCapturer* ScreenCapturer::Create() {
+ScreenCapturer* ScreenCapturer::Create(const DesktopCaptureOptions& context) {
scoped_ptr<ScreenCapturerMac> capturer(new ScreenCapturerMac());
if (!capturer->Init())
capturer.reset();
diff --git a/webrtc/modules/desktop_capture/screen_capturer_null.cc b/webrtc/modules/desktop_capture/screen_capturer_null.cc
index a94d603..a0bc7f1 100644
--- a/webrtc/modules/desktop_capture/screen_capturer_null.cc
+++ b/webrtc/modules/desktop_capture/screen_capturer_null.cc
@@ -13,20 +13,8 @@
namespace webrtc {
// static
-ScreenCapturer* ScreenCapturer::Create() {
+ScreenCapturer* ScreenCapturer::Create(const DesktopCaptureOptions& options) {
return NULL;
}
-#if defined(OS_LINUX)
-// static
-ScreenCapturer* ScreenCapturer::CreateWithXDamage(bool use_x_damage) {
- return NULL;
-}
-#elif defined(OS_WIN)
-// static
-ScreenCapturer* ScreenCapturer::CreateWithDisableAero(bool disable_aero) {
- return NULL;
-}
-#endif // defined(OS_WIN)
-
} // namespace webrtc
diff --git a/webrtc/modules/desktop_capture/screen_capturer_unittest.cc b/webrtc/modules/desktop_capture/screen_capturer_unittest.cc
index d5ba213..b4ae128 100644
--- a/webrtc/modules/desktop_capture/screen_capturer_unittest.cc
+++ b/webrtc/modules/desktop_capture/screen_capturer_unittest.cc
@@ -12,6 +12,7 @@
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
+#include "webrtc/modules/desktop_capture/desktop_capture_options.h"
#include "webrtc/modules/desktop_capture/desktop_frame.h"
#include "webrtc/modules/desktop_capture/desktop_region.h"
#include "webrtc/modules/desktop_capture/screen_capturer_mock_objects.h"
@@ -29,6 +30,11 @@
public:
SharedMemory* CreateSharedMemory(size_t size);
+ virtual void SetUp() OVERRIDE {
+ capturer_.reset(
+ ScreenCapturer::Create(DesktopCaptureOptions::CreateDefault()));
+ }
+
protected:
scoped_ptr<ScreenCapturer> capturer_;
MockMouseShapeObserver mouse_observer_;
@@ -54,7 +60,6 @@
}
TEST_F(ScreenCapturerTest, StartCapturer) {
- capturer_.reset(ScreenCapturer::Create());
capturer_->SetMouseShapeObserver(&mouse_observer_);
capturer_->Start(&callback_);
}
@@ -71,7 +76,6 @@
.Times(AnyNumber())
.WillRepeatedly(Return(static_cast<SharedMemory*>(NULL)));
- capturer_.reset(ScreenCapturer::Create());
capturer_->Start(&callback_);
capturer_->Capture(DesktopRegion());
@@ -106,7 +110,6 @@
.Times(AnyNumber())
.WillRepeatedly(Invoke(this, &ScreenCapturerTest::CreateSharedMemory));
- capturer_.reset(ScreenCapturer::Create());
capturer_->Start(&callback_);
capturer_->Capture(DesktopRegion());
diff --git a/webrtc/modules/desktop_capture/screen_capturer_win.cc b/webrtc/modules/desktop_capture/screen_capturer_win.cc
index 969b6a5..af9b18b 100644
--- a/webrtc/modules/desktop_capture/screen_capturer_win.cc
+++ b/webrtc/modules/desktop_capture/screen_capturer_win.cc
@@ -12,6 +12,7 @@
#include <windows.h>
+#include "webrtc/modules/desktop_capture/desktop_capture_options.h"
#include "webrtc/modules/desktop_capture/desktop_frame.h"
#include "webrtc/modules/desktop_capture/desktop_frame_win.h"
#include "webrtc/modules/desktop_capture/desktop_region.h"
@@ -43,7 +44,7 @@
// ScreenCapturerWin is double-buffered as required by ScreenCapturer.
class ScreenCapturerWin : public ScreenCapturer {
public:
- ScreenCapturerWin(bool disable_aero);
+ ScreenCapturerWin(const DesktopCaptureOptions& options);
virtual ~ScreenCapturerWin();
// Overridden from ScreenCapturer:
@@ -98,7 +99,7 @@
DISALLOW_COPY_AND_ASSIGN(ScreenCapturerWin);
};
-ScreenCapturerWin::ScreenCapturerWin(bool disable_aero)
+ScreenCapturerWin::ScreenCapturerWin(const DesktopCaptureOptions& options)
: callback_(NULL),
mouse_shape_observer_(NULL),
desktop_dc_(NULL),
@@ -106,7 +107,7 @@
dwmapi_library_(NULL),
composition_func_(NULL),
set_thread_execution_state_failed_(false) {
- if (disable_aero) {
+ if (options.disable_effects()) {
// Load dwmapi.dll dynamically since it is not available on XP.
if (!dwmapi_library_)
dwmapi_library_ = LoadLibrary(kDwmapiLibraryName);
@@ -353,13 +354,8 @@
} // namespace
// static
-ScreenCapturer* ScreenCapturer::Create() {
- return CreateWithDisableAero(true);
-}
-
-// static
-ScreenCapturer* ScreenCapturer::CreateWithDisableAero(bool disable_aero) {
- return new ScreenCapturerWin(disable_aero);
+ScreenCapturer* ScreenCapturer::Create(const DesktopCaptureOptions& options) {
+ return new ScreenCapturerWin(options);
}
} // namespace webrtc
diff --git a/webrtc/modules/desktop_capture/screen_capturer_x11.cc b/webrtc/modules/desktop_capture/screen_capturer_x11.cc
index 5cca65b..780d8cf 100644
--- a/webrtc/modules/desktop_capture/screen_capturer_x11.cc
+++ b/webrtc/modules/desktop_capture/screen_capturer_x11.cc
@@ -18,6 +18,7 @@
#include <X11/Xlib.h>
#include <X11/Xutil.h>
+#include "webrtc/modules/desktop_capture/desktop_capture_options.h"
#include "webrtc/modules/desktop_capture/desktop_frame.h"
#include "webrtc/modules/desktop_capture/differ.h"
#include "webrtc/modules/desktop_capture/mouse_cursor_shape.h"
@@ -46,7 +47,7 @@
virtual ~ScreenCapturerLinux();
// TODO(ajwong): Do we really want this to be synchronous?
- bool Init(bool use_x_damage);
+ bool Init(const DesktopCaptureOptions& options);
// DesktopCapturer interface.
virtual void Start(Callback* delegate) OVERRIDE;
@@ -57,6 +58,8 @@
MouseShapeObserver* mouse_shape_observer) OVERRIDE;
private:
+ Display* display() { return options_.x_display()->display(); }
+
void InitXDamage();
// Read and handle all currently-pending XEvents.
@@ -88,11 +91,12 @@
void DeinitXlib();
+ DesktopCaptureOptions options_;
+
Callback* callback_;
MouseShapeObserver* mouse_shape_observer_;
// X11 graphics context.
- Display* display_;
GC gc_;
Window root_window_;
@@ -131,7 +135,6 @@
ScreenCapturerLinux::ScreenCapturerLinux()
: callback_(NULL),
mouse_shape_observer_(NULL),
- display_(NULL),
gc_(NULL),
root_window_(BadValue),
has_xfixes_(false),
@@ -149,23 +152,17 @@
DeinitXlib();
}
-bool ScreenCapturerLinux::Init(bool use_x_damage) {
- // TODO(ajwong): We should specify the display string we are attaching to
- // in the constructor.
- display_ = XOpenDisplay(NULL);
- if (!display_) {
- LOG(LS_ERROR) << "Unable to open display";
- return false;
- }
+bool ScreenCapturerLinux::Init(const DesktopCaptureOptions& options) {
+ options_ = options;
- root_window_ = RootWindow(display_, DefaultScreen(display_));
+ root_window_ = RootWindow(display(), DefaultScreen(display()));
if (root_window_ == BadValue) {
LOG(LS_ERROR) << "Unable to get the root window";
DeinitXlib();
return false;
}
- gc_ = XCreateGC(display_, root_window_, 0, NULL);
+ gc_ = XCreateGC(display(), root_window_, 0, NULL);
if (gc_ == NULL) {
LOG(LS_ERROR) << "Unable to get graphics context";
DeinitXlib();
@@ -174,7 +171,7 @@
// Check for XFixes extension. This is required for cursor shape
// notifications, and for our use of XDamage.
- if (XFixesQueryExtension(display_, &xfixes_event_base_,
+ if (XFixesQueryExtension(display(), &xfixes_event_base_,
&xfixes_error_base_)) {
has_xfixes_ = true;
} else {
@@ -182,20 +179,20 @@
}
// Register for changes to the dimensions of the root window.
- XSelectInput(display_, root_window_, StructureNotifyMask);
+ XSelectInput(display(), root_window_, StructureNotifyMask);
- if (!x_server_pixel_buffer_.Init(display_, DefaultRootWindow(display_))) {
+ if (!x_server_pixel_buffer_.Init(display(), DefaultRootWindow(display()))) {
LOG(LS_ERROR) << "Failed to initialize pixel buffer.";
return false;
}
if (has_xfixes_) {
// Register for changes to the cursor shape.
- XFixesSelectCursorInput(display_, root_window_,
+ XFixesSelectCursorInput(display(), root_window_,
XFixesDisplayCursorNotifyMask);
}
- if (use_x_damage) {
+ if (options_.use_update_notifications()) {
InitXDamage();
}
@@ -209,7 +206,7 @@
}
// Check for XDamage extension.
- if (!XDamageQueryExtension(display_, &damage_event_base_,
+ if (!XDamageQueryExtension(display(), &damage_event_base_,
&damage_error_base_)) {
LOG(LS_INFO) << "X server does not support XDamage.";
return;
@@ -221,7 +218,7 @@
// properly.
// Request notifications every time the screen becomes damaged.
- damage_handle_ = XDamageCreate(display_, root_window_,
+ damage_handle_ = XDamageCreate(display(), root_window_,
XDamageReportNonEmpty);
if (!damage_handle_) {
LOG(LS_ERROR) << "Unable to initialize XDamage.";
@@ -229,9 +226,9 @@
}
// Create an XFixes server-side region to collate damage into.
- damage_region_ = XFixesCreateRegion(display_, 0, 0);
+ damage_region_ = XFixesCreateRegion(display(), 0, 0);
if (!damage_region_) {
- XDamageDestroy(display_, damage_handle_);
+ XDamageDestroy(display(), damage_handle_);
LOG(LS_ERROR) << "Unable to create XFixes region.";
return;
}
@@ -303,11 +300,11 @@
void ScreenCapturerLinux::ProcessPendingXEvents() {
// Find the number of events that are outstanding "now." We don't just loop
// on XPending because we want to guarantee this terminates.
- int events_to_process = XPending(display_);
+ int events_to_process = XPending(display());
XEvent e;
for (int i = 0; i < events_to_process; i++) {
- XNextEvent(display_, &e);
+ XNextEvent(display(), &e);
if (use_damage_ && (e.type == damage_event_base_ + XDamageNotify)) {
XDamageNotifyEvent* event = reinterpret_cast<XDamageNotifyEvent*>(&e);
DCHECK(event->level == XDamageReportNonEmpty);
@@ -329,7 +326,7 @@
void ScreenCapturerLinux::CaptureCursor() {
DCHECK(has_xfixes_);
- XFixesCursorImage* img = XFixesGetCursorImage(display_);
+ XFixesCursorImage* img = XFixesGetCursorImage(display());
if (!img) {
return;
}
@@ -375,10 +372,10 @@
x_server_pixel_buffer_.Synchronize();
if (use_damage_ && queue_.previous_frame()) {
// Atomically fetch and clear the damage region.
- XDamageSubtract(display_, damage_handle_, None, damage_region_);
+ XDamageSubtract(display(), damage_handle_, None, damage_region_);
int rects_num = 0;
XRectangle bounds;
- XRectangle* rects = XFixesFetchRegionAndBounds(display_, damage_region_,
+ XRectangle* rects = XFixesFetchRegionAndBounds(display(), damage_region_,
&rects_num, &bounds);
for (int i = 0; i < rects_num; ++i) {
updated_region->AddRect(DesktopRect::MakeXYWH(
@@ -430,7 +427,7 @@
queue_.Reset();
helper_.ClearInvalidRegion();
- if (!x_server_pixel_buffer_.Init(display_, DefaultRootWindow(display_))) {
+ if (!x_server_pixel_buffer_.Init(display(), DefaultRootWindow(display()))) {
LOG(LS_ERROR) << "Failed to initialize pixel buffer after screen "
"configuration change.";
}
@@ -465,38 +462,34 @@
void ScreenCapturerLinux::DeinitXlib() {
if (gc_) {
- XFreeGC(display_, gc_);
+ XFreeGC(display(), gc_);
gc_ = NULL;
}
x_server_pixel_buffer_.Release();
- if (display_) {
- if (damage_handle_)
- XDamageDestroy(display_, damage_handle_);
- if (damage_region_)
- XFixesDestroyRegion(display_, damage_region_);
- XCloseDisplay(display_);
- display_ = NULL;
- damage_handle_ = 0;
- damage_region_ = 0;
+ if (display()) {
+ if (damage_handle_) {
+ XDamageDestroy(display(), damage_handle_);
+ damage_handle_ = 0;
+ }
+
+ if (damage_region_) {
+ XFixesDestroyRegion(display(), damage_region_);
+ damage_region_ = 0;
+ }
}
}
} // namespace
// static
-ScreenCapturer* ScreenCapturer::Create() {
- scoped_ptr<ScreenCapturerLinux> capturer(new ScreenCapturerLinux());
- if (!capturer->Init(false))
- capturer.reset();
- return capturer.release();
-}
+ScreenCapturer* ScreenCapturer::Create(const DesktopCaptureOptions& options) {
+ if (!options.x_display())
+ return NULL;
-// static
-ScreenCapturer* ScreenCapturer::CreateWithXDamage(bool use_x_damage) {
scoped_ptr<ScreenCapturerLinux> capturer(new ScreenCapturerLinux());
- if (!capturer->Init(use_x_damage))
+ if (!capturer->Init(options))
capturer.reset();
return capturer.release();
}
diff --git a/webrtc/modules/desktop_capture/window_capturer.cc b/webrtc/modules/desktop_capture/window_capturer.cc
new file mode 100644
index 0000000..c5176d5
--- /dev/null
+++ b/webrtc/modules/desktop_capture/window_capturer.cc
@@ -0,0 +1,22 @@
+ /*
+ * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#include "webrtc/modules/desktop_capture/window_capturer.h"
+
+#include "webrtc/modules/desktop_capture/desktop_capture_options.h"
+
+namespace webrtc {
+
+// static
+WindowCapturer* WindowCapturer::Create() {
+ return Create(DesktopCaptureOptions::CreateDefault());
+}
+
+} // namespace webrtc
diff --git a/webrtc/modules/desktop_capture/window_capturer.h b/webrtc/modules/desktop_capture/window_capturer.h
index 8cc57c1..4e25c1b 100644
--- a/webrtc/modules/desktop_capture/window_capturer.h
+++ b/webrtc/modules/desktop_capture/window_capturer.h
@@ -20,6 +20,8 @@
namespace webrtc {
+class DesktopCaptureOptions;
+
class WindowCapturer : public DesktopCapturer {
public:
typedef intptr_t WindowId;
@@ -33,6 +35,9 @@
typedef std::vector<Window> WindowList;
+ static WindowCapturer* Create(const DesktopCaptureOptions& options);
+
+ // TODO(sergeyu): Remove this method. crbug.com/172183
static WindowCapturer* Create();
virtual ~WindowCapturer() {}
diff --git a/webrtc/modules/desktop_capture/window_capturer_mac.cc b/webrtc/modules/desktop_capture/window_capturer_mac.cc
index e78d95b..78f618d 100755
--- a/webrtc/modules/desktop_capture/window_capturer_mac.cc
+++ b/webrtc/modules/desktop_capture/window_capturer_mac.cc
@@ -196,7 +196,7 @@
} // namespace
// static
-WindowCapturer* WindowCapturer::Create() {
+WindowCapturer* WindowCapturer::Create(const DesktopCaptureOptions& options) {
return new WindowCapturerMac();
}
diff --git a/webrtc/modules/desktop_capture/window_capturer_null.cc b/webrtc/modules/desktop_capture/window_capturer_null.cc
index 8ea723a..7bb1247 100755
--- a/webrtc/modules/desktop_capture/window_capturer_null.cc
+++ b/webrtc/modules/desktop_capture/window_capturer_null.cc
@@ -69,7 +69,7 @@
} // namespace
// static
-WindowCapturer* WindowCapturer::Create() {
+WindowCapturer* WindowCapturer::Create(const DesktopCaptureOptions& options) {
return new WindowCapturerNull();
}
diff --git a/webrtc/modules/desktop_capture/window_capturer_unittest.cc b/webrtc/modules/desktop_capture/window_capturer_unittest.cc
index c6de16a..ad60a4b 100644
--- a/webrtc/modules/desktop_capture/window_capturer_unittest.cc
+++ b/webrtc/modules/desktop_capture/window_capturer_unittest.cc
@@ -10,9 +10,8 @@
#include "webrtc/modules/desktop_capture/window_capturer.h"
-#include <iostream>
-
#include "gtest/gtest.h"
+#include "webrtc/modules/desktop_capture/desktop_capture_options.h"
#include "webrtc/modules/desktop_capture/desktop_frame.h"
#include "webrtc/modules/desktop_capture/desktop_region.h"
#include "webrtc/system_wrappers/interface/logging.h"
@@ -24,7 +23,8 @@
public DesktopCapturer::Callback {
public:
void SetUp() OVERRIDE {
- capturer_.reset(WindowCapturer::Create());
+ capturer_.reset(
+ WindowCapturer::Create(DesktopCaptureOptions::CreateDefault()));
}
void TearDown() OVERRIDE {
diff --git a/webrtc/modules/desktop_capture/window_capturer_win.cc b/webrtc/modules/desktop_capture/window_capturer_win.cc
index 8fd380b..95f41db 100644
--- a/webrtc/modules/desktop_capture/window_capturer_win.cc
+++ b/webrtc/modules/desktop_capture/window_capturer_win.cc
@@ -248,7 +248,7 @@
} // namespace
// static
-WindowCapturer* WindowCapturer::Create() {
+WindowCapturer* WindowCapturer::Create(const DesktopCaptureOptions& options) {
return new WindowCapturerWin();
}
diff --git a/webrtc/modules/desktop_capture/window_capturer_x11.cc b/webrtc/modules/desktop_capture/window_capturer_x11.cc
index adafa94..5a14356 100755
--- a/webrtc/modules/desktop_capture/window_capturer_x11.cc
+++ b/webrtc/modules/desktop_capture/window_capturer_x11.cc
@@ -18,11 +18,14 @@
#include <algorithm>
#include <cassert>
+#include "webrtc/modules/desktop_capture/desktop_capture_options.h"
#include "webrtc/modules/desktop_capture/desktop_frame.h"
+#include "webrtc/modules/desktop_capture/x11/shared_x_display.h"
#include "webrtc/modules/desktop_capture/x11/x_error_trap.h"
#include "webrtc/modules/desktop_capture/x11/x_server_pixel_buffer.h"
#include "webrtc/system_wrappers/interface/logging.h"
#include "webrtc/system_wrappers/interface/scoped_ptr.h"
+#include "webrtc/system_wrappers/interface/scoped_refptr.h"
namespace webrtc {
@@ -83,7 +86,7 @@
class WindowCapturerLinux : public WindowCapturer {
public:
- WindowCapturerLinux();
+ WindowCapturerLinux(const DesktopCaptureOptions& options);
virtual ~WindowCapturerLinux();
// WindowCapturer interface.
@@ -95,6 +98,8 @@
virtual void Capture(const DesktopRegion& region) OVERRIDE;
private:
+ Display* display() { return x_display_->display(); }
+
// Iterates through |window| hierarchy to find first visible window, i.e. one
// that has WM_STATE property set to NormalState.
// See http://tronche.com/gui/x/icccm/sec-4.html#s-4.1.3.1 .
@@ -108,7 +113,7 @@
Callback* callback_;
- Display* display_;
+ scoped_refptr<SharedXDisplay> x_display_;
Atom wm_state_atom_;
Atom window_type_atom_;
@@ -121,26 +126,20 @@
DISALLOW_COPY_AND_ASSIGN(WindowCapturerLinux);
};
-WindowCapturerLinux::WindowCapturerLinux()
+WindowCapturerLinux::WindowCapturerLinux(const DesktopCaptureOptions& options)
: callback_(NULL),
- display_(NULL),
+ x_display_(options.x_display()),
has_composite_extension_(false),
selected_window_(0) {
- display_ = XOpenDisplay(NULL);
- if (!display_) {
- LOG(LS_ERROR) << "Failed to open display.";
- return;
- }
-
// Create Atoms so we don't need to do it every time they are used.
- wm_state_atom_ = XInternAtom(display_, "WM_STATE", True);
- window_type_atom_ = XInternAtom(display_, "_NET_WM_WINDOW_TYPE", True);
+ wm_state_atom_ = XInternAtom(display(), "WM_STATE", True);
+ window_type_atom_ = XInternAtom(display(), "_NET_WM_WINDOW_TYPE", True);
normal_window_type_atom_ = XInternAtom(
- display_, "_NET_WM_WINDOW_TYPE_NORMAL", True);
+ display(), "_NET_WM_WINDOW_TYPE_NORMAL", True);
int event_base, error_base, major_version, minor_version;
- if (XCompositeQueryExtension(display_, &event_base, &error_base) &&
- XCompositeQueryVersion(display_, &major_version, &minor_version) &&
+ if (XCompositeQueryExtension(display(), &event_base, &error_base) &&
+ XCompositeQueryVersion(display(), &major_version, &minor_version) &&
// XCompositeNameWindowPixmap() requires version 0.2
(major_version > 0 || minor_version >= 2)) {
has_composite_extension_ = true;
@@ -149,26 +148,20 @@
}
}
-WindowCapturerLinux::~WindowCapturerLinux() {
- if (display_)
- XCloseDisplay(display_);
-}
+WindowCapturerLinux::~WindowCapturerLinux() {}
bool WindowCapturerLinux::GetWindowList(WindowList* windows) {
- if (!display_)
- return false;
-
WindowList result;
- XErrorTrap error_trap(display_);
+ XErrorTrap error_trap(display());
- int num_screens = XScreenCount(display_);
+ int num_screens = XScreenCount(display());
for (int screen = 0; screen < num_screens; ++screen) {
- ::Window root_window = XRootWindow(display_, screen);
+ ::Window root_window = XRootWindow(display(), screen);
::Window parent;
::Window *children;
unsigned int num_children;
- int status = XQueryTree(display_, root_window, &root_window, &parent,
+ int status = XQueryTree(display(), root_window, &root_window, &parent,
&children, &num_children);
if (status == 0) {
LOG(LS_ERROR) << "Failed to query for child windows for screen "
@@ -198,7 +191,7 @@
}
bool WindowCapturerLinux::SelectWindow(WindowId id) {
- if (!x_server_pixel_buffer_.Init(display_, id))
+ if (!x_server_pixel_buffer_.Init(display(), id))
return false;
selected_window_ = id;
@@ -210,7 +203,7 @@
// Redirect drawing to an offscreen buffer (ie, turn on compositing). X11
// remembers who has requested this and will turn it off for us when we exit.
- XCompositeRedirectWindow(display_, id, CompositeRedirectAutomatic);
+ XCompositeRedirectWindow(display(), id, CompositeRedirectAutomatic);
return true;
}
@@ -244,7 +237,7 @@
::Window WindowCapturerLinux::GetApplicationWindow(::Window window) {
// Get WM_STATE property of the window.
- XWindowProperty<uint32_t> window_state(display_, window, wm_state_atom_);
+ XWindowProperty<uint32_t> window_state(display(), window, wm_state_atom_);
// WM_STATE is considered to be set to WithdrawnState when it missing.
int32_t state = window_state.is_valid() ?
@@ -262,7 +255,7 @@
::Window root, parent;
::Window *children;
unsigned int num_children;
- if (!XQueryTree(display_, window, &root, &parent, &children,
+ if (!XQueryTree(display(), window, &root, &parent, &children,
&num_children)) {
LOG(LS_ERROR) << "Failed to query for child windows although window"
<< "does not have a valid WM_STATE.";
@@ -289,7 +282,7 @@
// says this hint *should* be present on all windows, and we use the existence
// of _NET_WM_WINDOW_TYPE_NORMAL in the property to indicate a window is not
// a desktop element (that is, only "normal" windows should be shareable).
- XWindowProperty<uint32_t> window_type(display_, window, window_type_atom_);
+ XWindowProperty<uint32_t> window_type(display(), window, window_type_atom_);
if (window_type.is_valid() && window_type.size() > 0) {
uint32_t* end = window_type.data() + window_type.size();
bool is_normal = (end != std::find(
@@ -299,7 +292,7 @@
// Fall back on using the hint.
XClassHint class_hint;
- Status status = XGetClassHint(display_, window, &class_hint);
+ Status status = XGetClassHint(display(), window, &class_hint);
bool result = false;
if (status == 0) {
// No hints, assume this is a normal application window.
@@ -321,11 +314,11 @@
XTextProperty window_name;
window_name.value = NULL;
if (window) {
- status = XGetWMName(display_, window, &window_name);
+ status = XGetWMName(display(), window, &window_name);
if (status && window_name.value && window_name.nitems) {
int cnt;
char **list = NULL;
- status = Xutf8TextPropertyToTextList(display_, &window_name, &list,
+ status = Xutf8TextPropertyToTextList(display(), &window_name, &list,
&cnt);
if (status >= Success && cnt && *list) {
if (cnt > 1) {
@@ -347,8 +340,10 @@
} // namespace
// static
-WindowCapturer* WindowCapturer::Create() {
- return new WindowCapturerLinux();
+WindowCapturer* WindowCapturer::Create(const DesktopCaptureOptions& options) {
+ if (!options.x_display())
+ return NULL;
+ return new WindowCapturerLinux(options);
}
} // namespace webrtc
diff --git a/webrtc/modules/desktop_capture/x11/shared_x_display.cc b/webrtc/modules/desktop_capture/x11/shared_x_display.cc
new file mode 100644
index 0000000..c8f5a37
--- /dev/null
+++ b/webrtc/modules/desktop_capture/x11/shared_x_display.cc
@@ -0,0 +1,43 @@
+/*
+ * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#include "webrtc/modules/desktop_capture/x11/shared_x_display.h"
+
+#include "webrtc/system_wrappers/interface/logging.h"
+
+namespace webrtc {
+
+SharedXDisplay::SharedXDisplay(Display* display)
+ : display_(display) {
+ assert(display_);
+}
+
+SharedXDisplay::~SharedXDisplay() {
+ XCloseDisplay(display_);
+}
+
+// static
+scoped_refptr<SharedXDisplay> SharedXDisplay::Create(
+ const std::string& display_name) {
+ Display* display =
+ XOpenDisplay(display_name.empty() ? NULL : display_name.c_str());
+ if (!display) {
+ LOG(LS_ERROR) << "Unable to open display";
+ return NULL;
+ }
+ return new SharedXDisplay(display);
+}
+
+// static
+scoped_refptr<SharedXDisplay> SharedXDisplay::CreateDefault() {
+ return Create(std::string());
+}
+
+} // namespace webrtc
diff --git a/webrtc/modules/desktop_capture/x11/shared_x_display.h b/webrtc/modules/desktop_capture/x11/shared_x_display.h
new file mode 100644
index 0000000..93f6867
--- /dev/null
+++ b/webrtc/modules/desktop_capture/x11/shared_x_display.h
@@ -0,0 +1,58 @@
+/*
+ * Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
+ *
+ * Use of this source code is governed by a BSD-style license
+ * that can be found in the LICENSE file in the root of the source
+ * tree. An additional intellectual property rights grant can be found
+ * in the file PATENTS. All contributing project authors may
+ * be found in the AUTHORS file in the root of the source tree.
+ */
+
+#ifndef WEBRTC_MODULES_DESKTOP_CAPTURE_X11_SHARED_X_DISPLAY_H_
+#define WEBRTC_MODULES_DESKTOP_CAPTURE_X11_SHARED_X_DISPLAY_H_
+
+#include <assert.h>
+#include <X11/Xlib.h>
+
+#include <string>
+
+#include "webrtc/system_wrappers/interface/atomic32.h"
+#include "webrtc/system_wrappers/interface/scoped_refptr.h"
+
+namespace webrtc {
+
+// A ref-counted object to store XDisplay connection.
+class SharedXDisplay {
+ public:
+ // Takes ownership of |display|.
+ explicit SharedXDisplay(Display* display);
+
+ void AddRef() { ++ref_count_; }
+ void Release() {
+ if (--ref_count_ == 0)
+ delete this;
+ }
+
+ Display* display() { return display_; }
+
+ // Creates a new X11 Display for the |display_name|. NULL is returned if X11
+ // connection failed. Equivalent to CreateDefault() when |display_name| is
+ // empty.
+ static scoped_refptr<SharedXDisplay> Create(const std::string& display_name);
+
+ // Creates X11 Display connection for the default display (e.g. specified in
+ // DISPLAY). NULL is returned if X11 connection failed.
+ static scoped_refptr<SharedXDisplay> CreateDefault();
+
+ private:
+ ~SharedXDisplay();
+
+ Atomic32 ref_count_;
+ Display* display_;
+
+ DISALLOW_COPY_AND_ASSIGN(SharedXDisplay);
+};
+
+} // namespace webrtc
+
+#endif // WEBRTC_MODULES_DESKTOP_CAPTURE_X11_SHARED_X_DISPLAY_H_