blob: 90147a893d8a2d83e4a56cf436956a6f0557a392 [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 "ui/ozone/platform/drm/gpu/hardware_display_plane_manager.h"
#include <drm_fourcc.h>
#include <set>
#include <utility>
#include "base/logging.h"
#include "ui/gfx/geometry/rect.h"
#include "ui/ozone/platform/drm/gpu/drm_device.h"
#include "ui/ozone/platform/drm/gpu/scanout_buffer.h"
namespace ui {
namespace {
const float kFixedPointScaleValue = 65536.0f;
} // namespace
HardwareDisplayPlaneList::HardwareDisplayPlaneList() {
#if defined(USE_DRM_ATOMIC)
atomic_property_set.reset(drmModeAtomicAlloc());
#endif // defined(USE_DRM_ATOMIC)
}
HardwareDisplayPlaneList::~HardwareDisplayPlaneList() {
}
HardwareDisplayPlaneList::PageFlipInfo::PageFlipInfo(uint32_t crtc_id,
uint32_t framebuffer,
CrtcController* crtc)
: crtc_id(crtc_id), framebuffer(framebuffer), crtc(crtc) {
}
HardwareDisplayPlaneList::PageFlipInfo::PageFlipInfo(
const PageFlipInfo& other) = default;
HardwareDisplayPlaneList::PageFlipInfo::~PageFlipInfo() {
}
HardwareDisplayPlaneList::PageFlipInfo::Plane::Plane(int plane,
int framebuffer,
const gfx::Rect& bounds,
const gfx::Rect& src_rect)
: plane(plane),
framebuffer(framebuffer),
bounds(bounds),
src_rect(src_rect) {
}
HardwareDisplayPlaneList::PageFlipInfo::Plane::~Plane() {
}
HardwareDisplayPlaneManager::HardwareDisplayPlaneManager() : drm_(nullptr) {
}
HardwareDisplayPlaneManager::~HardwareDisplayPlaneManager() {
}
bool HardwareDisplayPlaneManager::Initialize(DrmDevice* drm) {
drm_ = drm;
// Try to get all of the planes if possible, so we don't have to try to
// discover hidden primary planes.
bool has_universal_planes = false;
#if defined(DRM_CLIENT_CAP_UNIVERSAL_PLANES)
has_universal_planes = drm->SetCapability(DRM_CLIENT_CAP_UNIVERSAL_PLANES, 1);
#endif // defined(DRM_CLIENT_CAP_UNIVERSAL_PLANES)
ScopedDrmResourcesPtr resources(drmModeGetResources(drm->get_fd()));
if (!resources) {
PLOG(ERROR) << "Failed to get resources";
return false;
}
ScopedDrmPlaneResPtr plane_resources(drmModeGetPlaneResources(drm->get_fd()));
if (!plane_resources) {
PLOG(ERROR) << "Failed to get plane resources";
return false;
}
crtcs_.clear();
for (int i = 0; i < resources->count_crtcs; ++i) {
crtcs_.push_back(resources->crtcs[i]);
}
uint32_t num_planes = plane_resources->count_planes;
std::set<uint32_t> plane_ids;
for (uint32_t i = 0; i < num_planes; ++i) {
ScopedDrmPlanePtr drm_plane(
drmModeGetPlane(drm->get_fd(), plane_resources->planes[i]));
if (!drm_plane) {
PLOG(ERROR) << "Failed to get plane " << i;
return false;
}
uint32_t formats_size = drm_plane->count_formats;
plane_ids.insert(drm_plane->plane_id);
std::unique_ptr<HardwareDisplayPlane> plane(
CreatePlane(drm_plane->plane_id, drm_plane->possible_crtcs));
std::vector<uint32_t> supported_formats(formats_size);
for (uint32_t j = 0; j < formats_size; j++)
supported_formats[j] = drm_plane->formats[j];
if (plane->Initialize(drm, supported_formats, false, false)) {
// CRTC controllers always assume they have a cursor plane and the cursor
// plane is updated via cursor specific DRM API. Hence, we dont keep
// track of Cursor plane here to avoid re-using it for any other purpose.
if (plane->type() != HardwareDisplayPlane::kCursor)
planes_.push_back(std::move(plane));
}
}
// crbug.com/464085: if driver reports no primary planes for a crtc, create a
// dummy plane for which we can assign exactly one overlay.
// TODO(dnicoara): refactor this to simplify AssignOverlayPlanes and move
// this workaround into HardwareDisplayPlaneLegacy.
if (!has_universal_planes) {
for (int i = 0; i < resources->count_crtcs; ++i) {
if (plane_ids.find(resources->crtcs[i] - 1) == plane_ids.end()) {
std::unique_ptr<HardwareDisplayPlane> dummy_plane(
CreatePlane(resources->crtcs[i] - 1, (1 << i)));
if (dummy_plane->Initialize(drm, std::vector<uint32_t>(), true,
false)) {
planes_.push_back(std::move(dummy_plane));
}
}
}
}
std::sort(planes_.begin(), planes_.end(),
[](const std::unique_ptr<HardwareDisplayPlane>& l,
const std::unique_ptr<HardwareDisplayPlane>& r) {
return l->plane_id() < r->plane_id();
});
PopulateSupportedFormats();
return true;
}
std::unique_ptr<HardwareDisplayPlane> HardwareDisplayPlaneManager::CreatePlane(
uint32_t plane_id,
uint32_t possible_crtcs) {
return std::unique_ptr<HardwareDisplayPlane>(
new HardwareDisplayPlane(plane_id, possible_crtcs));
}
HardwareDisplayPlane* HardwareDisplayPlaneManager::FindNextUnusedPlane(
size_t* index,
uint32_t crtc_index,
const OverlayPlane& overlay) const {
for (size_t i = *index; i < planes_.size(); ++i) {
auto plane = planes_[i].get();
if (!plane->in_use() && IsCompatible(plane, overlay, crtc_index)) {
*index = i + 1;
return plane;
}
}
return nullptr;
}
int HardwareDisplayPlaneManager::LookupCrtcIndex(uint32_t crtc_id) const {
for (size_t i = 0; i < crtcs_.size(); ++i)
if (crtcs_[i] == crtc_id)
return i;
return -1;
}
bool HardwareDisplayPlaneManager::IsCompatible(HardwareDisplayPlane* plane,
const OverlayPlane& overlay,
uint32_t crtc_index) const {
if (!plane->CanUseForCrtc(crtc_index))
return false;
if (!plane->IsSupportedFormat(overlay.buffer->GetFramebufferPixelFormat()))
return false;
// TODO(kalyank): We should check for z-order and any needed transformation
// support. Driver doesn't expose any property to check for z-order, can we
// rely on the sorting we do based on plane ids ?
return true;
}
void HardwareDisplayPlaneManager::PopulateSupportedFormats() {
std::set<uint32_t> supported_formats;
for (const auto& plane : planes_) {
const std::vector<uint32_t>& formats = plane->supported_formats();
supported_formats.insert(formats.begin(), formats.end());
}
supported_formats_.reserve(supported_formats.size());
supported_formats_.assign(supported_formats.begin(), supported_formats.end());
}
void HardwareDisplayPlaneManager::ResetCurrentPlaneList(
HardwareDisplayPlaneList* plane_list) const {
for (auto* hardware_plane : plane_list->plane_list) {
hardware_plane->set_in_use(false);
hardware_plane->set_owning_crtc(0);
}
plane_list->plane_list.clear();
plane_list->legacy_page_flips.clear();
#if defined(USE_DRM_ATOMIC)
plane_list->atomic_property_set.reset(drmModeAtomicAlloc());
#endif
}
void HardwareDisplayPlaneManager::BeginFrame(
HardwareDisplayPlaneList* plane_list) {
for (auto* plane : plane_list->old_plane_list) {
plane->set_in_use(false);
}
}
bool HardwareDisplayPlaneManager::AssignOverlayPlanes(
HardwareDisplayPlaneList* plane_list,
const OverlayPlaneList& overlay_list,
uint32_t crtc_id,
CrtcController* crtc) {
int crtc_index = LookupCrtcIndex(crtc_id);
if (crtc_index < 0) {
LOG(ERROR) << "Cannot find crtc " << crtc_id;
return false;
}
size_t plane_idx = 0;
HardwareDisplayPlane* primary_plane = nullptr;
gfx::Rect primary_display_bounds;
uint32_t primary_format;
for (const auto& plane : overlay_list) {
HardwareDisplayPlane* hw_plane =
FindNextUnusedPlane(&plane_idx, crtc_index, plane);
if (!hw_plane) {
LOG(ERROR) << "Failed to find a free plane for crtc " << crtc_id;
ResetCurrentPlaneList(plane_list);
return false;
}
gfx::Rect fixed_point_rect;
uint32_t fourcc_format = plane.buffer->GetFramebufferPixelFormat();
if (hw_plane->type() != HardwareDisplayPlane::kDummy) {
const gfx::Size& size = plane.buffer->GetSize();
gfx::RectF crop_rect = plane.crop_rect;
crop_rect.Scale(size.width(), size.height());
// This returns a number in 16.16 fixed point, required by the DRM overlay
// APIs.
auto to_fixed_point =
[](double v) -> uint32_t { return v * kFixedPointScaleValue; };
fixed_point_rect = gfx::Rect(to_fixed_point(crop_rect.x()),
to_fixed_point(crop_rect.y()),
to_fixed_point(crop_rect.width()),
to_fixed_point(crop_rect.height()));
}
// If Overlay completely covers primary and isn't transparent, than use
// it as primary. This reduces the no of planes which need to be read in
// display controller side.
if (primary_plane) {
bool needs_blending = true;
if (fourcc_format == DRM_FORMAT_XRGB8888)
needs_blending = false;
// TODO(kalyank): Check if we can move this optimization to
// DrmOverlayCandidatesHost.
if (!needs_blending && primary_format == fourcc_format &&
primary_display_bounds == plane.display_bounds) {
ResetCurrentPlaneList(plane_list);
hw_plane = primary_plane;
}
} else {
primary_plane = hw_plane;
primary_display_bounds = plane.display_bounds;
primary_format = fourcc_format;
}
if (!SetPlaneData(plane_list, hw_plane, plane, crtc_id, fixed_point_rect,
crtc)) {
ResetCurrentPlaneList(plane_list);
return false;
}
plane_list->plane_list.push_back(hw_plane);
hw_plane->set_owning_crtc(crtc_id);
hw_plane->set_in_use(true);
}
return true;
}
const std::vector<uint32_t>& HardwareDisplayPlaneManager::GetSupportedFormats()
const {
return supported_formats_;
}
bool HardwareDisplayPlaneManager::IsFormatSupported(uint32_t fourcc_format,
uint32_t z_order,
uint32_t crtc_id) const {
bool format_supported = false;
int crtc_index = LookupCrtcIndex(crtc_id);
if (crtc_index < 0) {
LOG(ERROR) << "Cannot find crtc " << crtc_id;
return format_supported;
}
// We dont have a way to query z_order of a plane. This is a temporary
// solution till driver exposes z_order property.
uint32_t plane_z_order = 0;
for (const auto& hardware_plane : planes_) {
if (plane_z_order > z_order)
break;
if (!hardware_plane->CanUseForCrtc(crtc_index))
continue;
if (plane_z_order == z_order) {
if (hardware_plane->IsSupportedFormat(fourcc_format))
format_supported = true;
break;
} else {
plane_z_order++;
}
}
return format_supported;
}
} // namespace ui