| Index: ui/ozone/platform/drm/gpu/hardware_display_controller_unittest.cc
|
| diff --git a/ui/ozone/platform/drm/gpu/hardware_display_controller_unittest.cc b/ui/ozone/platform/drm/gpu/hardware_display_controller_unittest.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..eaa3cdc25be27b9b2083627561a0ca9b6b1aca1b
|
| --- /dev/null
|
| +++ b/ui/ozone/platform/drm/gpu/hardware_display_controller_unittest.cc
|
| @@ -0,0 +1,331 @@
|
| +// Copyright 2014 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 "base/bind.h"
|
| +#include "testing/gtest/include/gtest/gtest.h"
|
| +#include "third_party/skia/include/core/SkCanvas.h"
|
| +#include "ui/ozone/platform/drm/gpu/crtc_controller.h"
|
| +#include "ui/ozone/platform/drm/gpu/drm_buffer.h"
|
| +#include "ui/ozone/platform/drm/gpu/drm_device.h"
|
| +#include "ui/ozone/platform/drm/gpu/hardware_display_controller.h"
|
| +#include "ui/ozone/platform/drm/test/mock_drm_device.h"
|
| +#include "ui/ozone/public/native_pixmap.h"
|
| +
|
| +namespace {
|
| +
|
| +// Create a basic mode for a 6x4 screen.
|
| +const drmModeModeInfo kDefaultMode =
|
| + {0, 6, 0, 0, 0, 0, 4, 0, 0, 0, 0, 0, 0, 0, {'\0'}};
|
| +
|
| +const uint32_t kPrimaryCrtc = 1;
|
| +const uint32_t kPrimaryConnector = 2;
|
| +const uint32_t kSecondaryCrtc = 3;
|
| +const uint32_t kSecondaryConnector = 4;
|
| +const size_t kPlanesPerCrtc = 2;
|
| +
|
| +const gfx::Size kDefaultModeSize(kDefaultMode.hdisplay, kDefaultMode.vdisplay);
|
| +const gfx::SizeF kDefaultModeSizeF(1.0, 1.0);
|
| +
|
| +class MockScanoutBuffer : public ui::ScanoutBuffer {
|
| + public:
|
| + MockScanoutBuffer(const gfx::Size& size) : size_(size) {}
|
| +
|
| + // ScanoutBuffer:
|
| + uint32_t GetFramebufferId() const override { return 0; }
|
| + uint32_t GetHandle() const override { return 0; }
|
| + gfx::Size GetSize() const override { return size_; }
|
| +
|
| + private:
|
| + ~MockScanoutBuffer() override {}
|
| +
|
| + gfx::Size size_;
|
| +
|
| + DISALLOW_COPY_AND_ASSIGN(MockScanoutBuffer);
|
| +};
|
| +
|
| +} // namespace
|
| +
|
| +class HardwareDisplayControllerTest : public testing::Test {
|
| + public:
|
| + HardwareDisplayControllerTest() : page_flips_(0) {}
|
| + ~HardwareDisplayControllerTest() override {}
|
| +
|
| + void SetUp() override;
|
| + void TearDown() override;
|
| +
|
| + void PageFlipCallback(gfx::SwapResult);
|
| +
|
| + protected:
|
| + scoped_ptr<ui::HardwareDisplayController> controller_;
|
| + scoped_refptr<ui::MockDrmDevice> drm_;
|
| +
|
| + int page_flips_;
|
| +
|
| + private:
|
| + DISALLOW_COPY_AND_ASSIGN(HardwareDisplayControllerTest);
|
| +};
|
| +
|
| +void HardwareDisplayControllerTest::SetUp() {
|
| + std::vector<uint32_t> crtcs;
|
| + crtcs.push_back(kPrimaryCrtc);
|
| + crtcs.push_back(kSecondaryCrtc);
|
| + drm_ = new ui::MockDrmDevice(false, crtcs, kPlanesPerCrtc);
|
| + controller_.reset(new ui::HardwareDisplayController(
|
| + scoped_ptr<ui::CrtcController>(
|
| + new ui::CrtcController(drm_.get(), kPrimaryCrtc, kPrimaryConnector)),
|
| + gfx::Point()));
|
| +}
|
| +
|
| +void HardwareDisplayControllerTest::TearDown() {
|
| + controller_.reset();
|
| + drm_ = nullptr;
|
| +}
|
| +
|
| +void HardwareDisplayControllerTest::PageFlipCallback(gfx::SwapResult) {
|
| + page_flips_++;
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, CheckModesettingResult) {
|
| + ui::OverlayPlane plane(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| +
|
| + EXPECT_TRUE(controller_->Modeset(plane, kDefaultMode));
|
| + EXPECT_FALSE(plane.buffer->HasOneRef());
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, CheckStateAfterPageFlip) {
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| +
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| +
|
| + ui::OverlayPlane plane2(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + std::vector<ui::OverlayPlane> planes =
|
| + std::vector<ui::OverlayPlane>(1, plane2);
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| + drm_->RunCallbacks();
|
| + EXPECT_TRUE(plane1.buffer->HasOneRef());
|
| + EXPECT_FALSE(plane2.buffer->HasOneRef());
|
| +
|
| + EXPECT_EQ(1, drm_->get_page_flip_call_count());
|
| + EXPECT_EQ(0, drm_->get_overlay_flip_call_count());
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, CheckStateIfModesetFails) {
|
| + drm_->set_set_crtc_expectation(false);
|
| +
|
| + ui::OverlayPlane plane(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| +
|
| + EXPECT_FALSE(controller_->Modeset(plane, kDefaultMode));
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, CheckStateIfPageFlipFails) {
|
| + drm_->set_page_flip_expectation(false);
|
| +
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| +
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| +
|
| + ui::OverlayPlane plane2(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + std::vector<ui::OverlayPlane> planes =
|
| + std::vector<ui::OverlayPlane>(1, plane2);
|
| + EXPECT_FALSE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| + drm_->RunCallbacks();
|
| + planes.clear();
|
| +
|
| + EXPECT_FALSE(plane1.buffer->HasOneRef());
|
| + EXPECT_TRUE(plane2.buffer->HasOneRef());
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, CheckOverlayPresent) {
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + ui::OverlayPlane plane2(
|
| + scoped_refptr<ui::ScanoutBuffer>(new MockScanoutBuffer(kDefaultModeSize)),
|
| + 1, gfx::OVERLAY_TRANSFORM_NONE, gfx::Rect(kDefaultModeSize),
|
| + gfx::RectF(kDefaultModeSizeF));
|
| +
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| +
|
| + std::vector<ui::OverlayPlane> planes;
|
| + planes.push_back(plane1);
|
| + planes.push_back(plane2);
|
| +
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| + drm_->RunCallbacks();
|
| + EXPECT_EQ(1, drm_->get_page_flip_call_count());
|
| + EXPECT_EQ(1, drm_->get_overlay_flip_call_count());
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, CheckOverlayTestMode) {
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + ui::OverlayPlane plane2(
|
| + scoped_refptr<ui::ScanoutBuffer>(new MockScanoutBuffer(kDefaultModeSize)),
|
| + 1, gfx::OVERLAY_TRANSFORM_NONE, gfx::Rect(kDefaultModeSize),
|
| + gfx::RectF(kDefaultModeSizeF));
|
| +
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| +
|
| + std::vector<ui::OverlayPlane> planes;
|
| + planes.push_back(plane1);
|
| + planes.push_back(plane2);
|
| +
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| + drm_->RunCallbacks();
|
| + EXPECT_EQ(1, drm_->get_page_flip_call_count());
|
| + EXPECT_EQ(1, drm_->get_overlay_flip_call_count());
|
| +
|
| + // A test call shouldn't cause new flips, but should succeed.
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, true,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| + drm_->RunCallbacks();
|
| + EXPECT_EQ(1, drm_->get_page_flip_call_count());
|
| + EXPECT_EQ(1, drm_->get_overlay_flip_call_count());
|
| +
|
| + // Regular flips should continue on normally.
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| + drm_->RunCallbacks();
|
| + EXPECT_EQ(2, drm_->get_page_flip_call_count());
|
| + EXPECT_EQ(2, drm_->get_overlay_flip_call_count());
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, RejectUnderlays) {
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + ui::OverlayPlane plane2(
|
| + scoped_refptr<ui::ScanoutBuffer>(new MockScanoutBuffer(kDefaultModeSize)),
|
| + -1, gfx::OVERLAY_TRANSFORM_NONE, gfx::Rect(kDefaultModeSize),
|
| + gfx::RectF(kDefaultModeSizeF));
|
| +
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| +
|
| + std::vector<ui::OverlayPlane> planes;
|
| + planes.push_back(plane1);
|
| + planes.push_back(plane2);
|
| +
|
| + EXPECT_FALSE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, PageflipMirroredControllers) {
|
| + controller_->AddCrtc(scoped_ptr<ui::CrtcController>(
|
| + new ui::CrtcController(drm_.get(), kSecondaryCrtc, kSecondaryConnector)));
|
| +
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| + EXPECT_EQ(2, drm_->get_set_crtc_call_count());
|
| +
|
| + ui::OverlayPlane plane2(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + std::vector<ui::OverlayPlane> planes =
|
| + std::vector<ui::OverlayPlane>(1, plane2);
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| + drm_->RunCallbacks();
|
| + EXPECT_EQ(2, drm_->get_page_flip_call_count());
|
| + EXPECT_EQ(1, page_flips_);
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, PlaneStateAfterRemoveCrtc) {
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| + std::vector<ui::OverlayPlane> planes =
|
| + std::vector<ui::OverlayPlane>(1, plane1);
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| + drm_->RunCallbacks();
|
| +
|
| + const ui::HardwareDisplayPlane* owned_plane = nullptr;
|
| + for (const auto& plane : drm_->plane_manager()->planes())
|
| + if (plane->in_use())
|
| + owned_plane = plane;
|
| + ASSERT_TRUE(owned_plane != nullptr);
|
| + EXPECT_EQ(kPrimaryCrtc, owned_plane->owning_crtc());
|
| + // Removing the crtc should free the plane.
|
| + scoped_ptr<ui::CrtcController> crtc =
|
| + controller_->RemoveCrtc(drm_, kPrimaryCrtc);
|
| + EXPECT_FALSE(owned_plane->in_use());
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, ModesetWhilePageFlipping) {
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| + std::vector<ui::OverlayPlane> planes =
|
| + std::vector<ui::OverlayPlane>(1, plane1);
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| +
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| + drm_->RunCallbacks();
|
| + EXPECT_EQ(1, page_flips_);
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, AddCrtcMidPageFlip) {
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| + std::vector<ui::OverlayPlane> planes =
|
| + std::vector<ui::OverlayPlane>(1, plane1);
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| +
|
| + controller_->AddCrtc(scoped_ptr<ui::CrtcController>(
|
| + new ui::CrtcController(drm_.get(), kSecondaryCrtc, kSecondaryConnector)));
|
| +
|
| + drm_->RunCallbacks();
|
| + EXPECT_EQ(1, page_flips_);
|
| +}
|
| +
|
| +TEST_F(HardwareDisplayControllerTest, RemoveCrtcMidPageFlip) {
|
| + ui::OverlayPlane plane1(scoped_refptr<ui::ScanoutBuffer>(
|
| + new MockScanoutBuffer(kDefaultModeSize)));
|
| + EXPECT_TRUE(controller_->Modeset(plane1, kDefaultMode));
|
| + std::vector<ui::OverlayPlane> planes =
|
| + std::vector<ui::OverlayPlane>(1, plane1);
|
| + EXPECT_TRUE(controller_->SchedulePageFlip(
|
| + planes, false, false,
|
| + base::Bind(&HardwareDisplayControllerTest::PageFlipCallback,
|
| + base::Unretained(this))));
|
| +
|
| + controller_->RemoveCrtc(drm_, kPrimaryCrtc);
|
| +
|
| + EXPECT_EQ(1, page_flips_);
|
| + drm_->RunCallbacks();
|
| + EXPECT_EQ(1, page_flips_);
|
| +}
|
|
|