blob: eaa3cdc25be27b9b2083627561a0ca9b6b1aca1b [file] [log] [blame]
// 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_);
}