webrtc_m130/modules/desktop_capture/screen_drawer_unittest.cc

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

164 lines
5.5 KiB
C++
Raw Normal View History

/*
* Copyright (c) 2016 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 "modules/desktop_capture/screen_drawer.h"
#include <stdint.h>
#include <atomic>
#include "absl/memory/memory.h"
#include "api/function_view.h"
#include "rtc_base/checks.h"
#include "rtc_base/logging.h"
#include "rtc_base/platform_thread.h"
#include "rtc_base/random.h"
#include "rtc_base/time_utils.h"
#include "system_wrappers/include/sleep.h"
#include "test/gtest.h"
#if defined(WEBRTC_POSIX)
#include "modules/desktop_capture/screen_drawer_lock_posix.h"
#endif
namespace webrtc {
namespace {
void TestScreenDrawerLock(
rtc::FunctionView<std::unique_ptr<ScreenDrawerLock>()> ctor) {
constexpr int kLockDurationMs = 100;
RTC_DCHECK(ctor);
std::atomic<bool> created(false);
std::atomic<bool> ready(false);
class Task {
public:
Task(std::atomic<bool>* created,
const std::atomic<bool>& ready,
rtc::FunctionView<std::unique_ptr<ScreenDrawerLock>()> ctor)
: created_(created), ready_(ready), ctor_(ctor) {}
~Task() = default;
static void RunTask(void* me) {
Task* task = static_cast<Task*>(me);
std::unique_ptr<ScreenDrawerLock> lock = task->ctor_();
ASSERT_TRUE(!!lock);
task->created_->store(true);
// Wait for the main thread to get the signal of created_.
while (!task->ready_.load()) {
SleepMs(1);
}
// At this point, main thread should begin to create a second lock. Though
// it's still possible the second lock won't be created before the
// following sleep has been finished, the possibility will be
// significantly reduced.
const int64_t current_ms = rtc::TimeMillis();
// SleepMs() may return early. See
// https://cs.chromium.org/chromium/src/third_party/webrtc/system_wrappers/include/sleep.h?rcl=4a604c80cecce18aff6fc5e16296d04675312d83&l=20
// But we need to ensure at least 100 ms has been passed before unlocking
// |lock|.
while (rtc::TimeMillis() - current_ms < kLockDurationMs) {
SleepMs(kLockDurationMs - (rtc::TimeMillis() - current_ms));
}
}
private:
std::atomic<bool>* const created_;
const std::atomic<bool>& ready_;
const rtc::FunctionView<std::unique_ptr<ScreenDrawerLock>()> ctor_;
} task(&created, ready, ctor);
rtc::PlatformThread lock_thread(&Task::RunTask, &task, "lock_thread");
lock_thread.Start();
// Wait for the first lock in Task::RunTask() to be created.
// TODO(zijiehe): Find a better solution to wait for the creation of the first
// lock. See
// https://chromium-review.googlesource.com/c/607688/13/webrtc/modules/desktop_capture/screen_drawer_unittest.cc
while (!created.load()) {
SleepMs(1);
}
const int64_t start_ms = rtc::TimeMillis();
ready.store(true);
// This is unlikely to fail, but just in case current thread is too laggy and
// cause the SleepMs() in RunTask() to finish before we creating another lock.
ASSERT_GT(kLockDurationMs, rtc::TimeMillis() - start_ms);
ctor();
ASSERT_LE(kLockDurationMs, rtc::TimeMillis() - start_ms);
lock_thread.Stop();
}
} // namespace
// These are a set of manual test cases, as we do not have an automatical way to
// detect whether a ScreenDrawer on a certain platform works well without
// ScreenCapturer(s). So you may execute these test cases with
// --gtest_also_run_disabled_tests --gtest_filter=ScreenDrawerTest.*.
TEST(ScreenDrawerTest, DISABLED_DrawRectangles) {
std::unique_ptr<ScreenDrawer> drawer = ScreenDrawer::Create();
if (!drawer) {
RTC_LOG(LS_WARNING)
<< "No ScreenDrawer implementation for current platform.";
Reland of [WebRTC] A real ScreenCapturer test (patchset #1 id:1 of https://codereview.webrtc.org/2310953002/ ) Reason for revert: Resubmit capturer tests Original issue's description: > Revert of [WebRTC] A real ScreenCapturer test (patchset #8 id:240001 of https://codereview.webrtc.org/2268093002/ ) > > Reason for revert: > ScreenCapturerTest.CaptureUpdatedRegion fails on Win DrMemory Full. > > Original issue's description: > > [WebRTC] A real ScreenCapturer test > > > > We do not have a real ScreenCapturer test before. And after CL 2210443002, a new > > ScreenDrawer interface is added to the code base to draw various shapes on the > > screen. This change is to use ScreenDrawer to test ScreenCapturer. Besides test > > cases, some other changes are included, > > > > 1. A WaitForPendingPaintings() function in ScreenDrawer, to wait for a > > ScreenDrawer to finish all the pending draws. This function now only sleeps 50 > > milliseconds on X11 and 100 milliseconds on Windows. > > > > 2. A Color structure to help handle a big-endian or little-endian safe color and > > provide functions to compare with DesktopFrame::data(). Both ScreenDrawer and > > DesktopFrameGenerator (in change 2202443002) can use this class to create colors > > and compare with or paint to a DesktopFrame. > > > > 3. ScreenDrawer now uses Color structure instead of uint32_t. > > > > BUG=314516 > > > > TBR=kjellander@chromium.org > > > > Committed: https://crrev.com/9d1c54ace0dc9f68da0152aa1ded2a8dba0a43ae > > Cr-Commit-Position: refs/heads/master@{#14058} > > TBR=sergeyu@chromium.org,jamiewalch@chromium.org,kjellander@chromium.org,zijiehe@chromium.org > # Not skipping CQ checks because original CL landed more than 1 days ago. > BUG=314516 > > Committed: https://crrev.com/4c44202dc348613695a4b529bbd7c9bdab6195ec > Cr-Commit-Position: refs/heads/master@{#14071} TBR=sergeyu@chromium.org,jamiewalch@chromium.org,kjellander@chromium.org,asapersson@webrtc.org # Skipping CQ checks because original CL landed less than 1 days ago. NOPRESUBMIT=true NOTREECHECKS=true NOTRY=true BUG=644130 Review-Url: https://codereview.webrtc.org/2313653003 Cr-Commit-Position: refs/heads/master@{#14113}
2016-09-07 11:52:25 -07:00
return;
}
if (drawer->DrawableRegion().is_empty()) {
RTC_LOG(LS_WARNING)
<< "ScreenDrawer of current platform does not provide a "
"non-empty DrawableRegion().";
return;
}
DesktopRect rect = drawer->DrawableRegion();
Random random(rtc::TimeMicros());
for (int i = 0; i < 100; i++) {
// Make sure we at least draw one pixel.
int left = random.Rand(rect.left(), rect.right() - 2);
int top = random.Rand(rect.top(), rect.bottom() - 2);
drawer->DrawRectangle(
DesktopRect::MakeLTRB(left, top, random.Rand(left + 1, rect.right()),
random.Rand(top + 1, rect.bottom())),
Reland of [WebRTC] A real ScreenCapturer test (patchset #1 id:1 of https://codereview.webrtc.org/2310953002/ ) Reason for revert: Resubmit capturer tests Original issue's description: > Revert of [WebRTC] A real ScreenCapturer test (patchset #8 id:240001 of https://codereview.webrtc.org/2268093002/ ) > > Reason for revert: > ScreenCapturerTest.CaptureUpdatedRegion fails on Win DrMemory Full. > > Original issue's description: > > [WebRTC] A real ScreenCapturer test > > > > We do not have a real ScreenCapturer test before. And after CL 2210443002, a new > > ScreenDrawer interface is added to the code base to draw various shapes on the > > screen. This change is to use ScreenDrawer to test ScreenCapturer. Besides test > > cases, some other changes are included, > > > > 1. A WaitForPendingPaintings() function in ScreenDrawer, to wait for a > > ScreenDrawer to finish all the pending draws. This function now only sleeps 50 > > milliseconds on X11 and 100 milliseconds on Windows. > > > > 2. A Color structure to help handle a big-endian or little-endian safe color and > > provide functions to compare with DesktopFrame::data(). Both ScreenDrawer and > > DesktopFrameGenerator (in change 2202443002) can use this class to create colors > > and compare with or paint to a DesktopFrame. > > > > 3. ScreenDrawer now uses Color structure instead of uint32_t. > > > > BUG=314516 > > > > TBR=kjellander@chromium.org > > > > Committed: https://crrev.com/9d1c54ace0dc9f68da0152aa1ded2a8dba0a43ae > > Cr-Commit-Position: refs/heads/master@{#14058} > > TBR=sergeyu@chromium.org,jamiewalch@chromium.org,kjellander@chromium.org,zijiehe@chromium.org > # Not skipping CQ checks because original CL landed more than 1 days ago. > BUG=314516 > > Committed: https://crrev.com/4c44202dc348613695a4b529bbd7c9bdab6195ec > Cr-Commit-Position: refs/heads/master@{#14071} TBR=sergeyu@chromium.org,jamiewalch@chromium.org,kjellander@chromium.org,asapersson@webrtc.org # Skipping CQ checks because original CL landed less than 1 days ago. NOPRESUBMIT=true NOTREECHECKS=true NOTRY=true BUG=644130 Review-Url: https://codereview.webrtc.org/2313653003 Cr-Commit-Position: refs/heads/master@{#14113}
2016-09-07 11:52:25 -07:00
RgbaColor(random.Rand<uint8_t>(), random.Rand<uint8_t>(),
random.Rand<uint8_t>(), random.Rand<uint8_t>()));
if (i == 50) {
SleepMs(10000);
}
}
SleepMs(10000);
}
#if defined(THREAD_SANITIZER) // bugs.webrtc.org/10019
#define MAYBE_TwoScreenDrawerLocks DISABLED_TwoScreenDrawerLocks
#else
#define MAYBE_TwoScreenDrawerLocks TwoScreenDrawerLocks
#endif
TEST(ScreenDrawerTest, MAYBE_TwoScreenDrawerLocks) {
#if defined(WEBRTC_POSIX)
// ScreenDrawerLockPosix won't be able to unlink the named semaphore. So use a
// different semaphore name here to avoid deadlock.
const char* semaphore_name = "GSDL8784541a812011e788ff67427b";
ScreenDrawerLockPosix::Unlink(semaphore_name);
TestScreenDrawerLock([semaphore_name]() {
return absl::make_unique<ScreenDrawerLockPosix>(semaphore_name);
});
#elif defined(WEBRTC_WIN)
TestScreenDrawerLock([]() { return ScreenDrawerLock::Create(); });
#endif
}
} // namespace webrtc