Index: ui/ozone/platform/drm/gpu/hardware_display_plane_manager.cc |
diff --git a/ui/ozone/platform/drm/gpu/hardware_display_plane_manager.cc b/ui/ozone/platform/drm/gpu/hardware_display_plane_manager.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..a790d6219ff9956fe8849be94c78d2ba4bde7ac4 |
--- /dev/null |
+++ b/ui/ozone/platform/drm/gpu/hardware_display_plane_manager.cc |
@@ -0,0 +1,235 @@ |
+// 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.h> |
+#include <xf86drm.h> |
+ |
+#include <set> |
+ |
+#include "base/logging.h" |
+#include "ui/gfx/geometry/rect.h" |
+#include "ui/ozone/platform/drm/gpu/crtc_controller.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/gpu/scanout_buffer.h" |
+#include "ui/ozone/public/ozone_switches.h" |
+ |
+namespace ui { |
+namespace { |
+ |
+const float kFixedPointScaleValue = 65536.0f; |
+ |
+} // namespace |
+ |
+HardwareDisplayPlaneList::HardwareDisplayPlaneList() { |
+#if defined(USE_DRM_ATOMIC) |
+ atomic_property_set.reset(drmModePropertySetAlloc()); |
+#endif // defined(USE_DRM_ATOMIC) |
+} |
+ |
+HardwareDisplayPlaneList::~HardwareDisplayPlaneList() { |
+ for (auto* plane : plane_list) { |
+ plane->set_in_use(false); |
+ plane->set_owning_crtc(0); |
+ } |
+ for (auto* plane : old_plane_list) { |
+ plane->set_in_use(false); |
+ plane->set_owning_crtc(0); |
+ } |
+} |
+ |
+HardwareDisplayPlaneList::PageFlipInfo::PageFlipInfo(uint32_t crtc_id, |
+ uint32_t framebuffer, |
+ CrtcController* crtc) |
+ : crtc_id(crtc_id), framebuffer(framebuffer), crtc(crtc) { |
+} |
+ |
+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; |
+ } |
+ plane_ids.insert(drm_plane->plane_id); |
+ scoped_ptr<HardwareDisplayPlane> plane( |
+ CreatePlane(drm_plane->plane_id, drm_plane->possible_crtcs)); |
+ if (plane->Initialize(drm)) |
+ planes_.push_back(plane.Pass()); |
+ } |
+ |
+ // 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()) { |
+ scoped_ptr<HardwareDisplayPlane> dummy_plane( |
+ CreatePlane(resources->crtcs[i] - 1, (1 << i))); |
+ dummy_plane->set_is_dummy(true); |
+ if (dummy_plane->Initialize(drm)) |
+ planes_.push_back(dummy_plane.Pass()); |
+ } |
+ } |
+ } |
+ |
+ std::sort(planes_.begin(), planes_.end(), |
+ [](HardwareDisplayPlane* l, HardwareDisplayPlane* r) { |
+ return l->plane_id() < r->plane_id(); |
+ }); |
+ return true; |
+} |
+ |
+scoped_ptr<HardwareDisplayPlane> HardwareDisplayPlaneManager::CreatePlane( |
+ uint32_t plane_id, |
+ uint32_t possible_crtcs) { |
+ return scoped_ptr<HardwareDisplayPlane>( |
+ new HardwareDisplayPlane(plane_id, possible_crtcs)); |
+} |
+ |
+HardwareDisplayPlane* HardwareDisplayPlaneManager::FindNextUnusedPlane( |
+ size_t* index, |
+ uint32_t crtc_index) { |
+ for (size_t i = *index; i < planes_.size(); ++i) { |
+ auto plane = planes_[i]; |
+ if (!plane->in_use() && plane->CanUseForCrtc(crtc_index)) { |
+ *index = i + 1; |
+ return plane; |
+ } |
+ } |
+ return nullptr; |
+} |
+ |
+int HardwareDisplayPlaneManager::LookupCrtcIndex(uint32_t crtc_id) { |
+ for (size_t i = 0; i < crtcs_.size(); ++i) |
+ if (crtcs_[i] == crtc_id) |
+ return i; |
+ return -1; |
+} |
+ |
+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; |
+ for (const auto& plane : overlay_list) { |
+ HardwareDisplayPlane* hw_plane = |
+ FindNextUnusedPlane(&plane_idx, crtc_index); |
+ if (!hw_plane) { |
+ LOG(ERROR) << "Failed to find a free plane for crtc " << crtc_id; |
+ return false; |
+ } |
+ |
+ gfx::Rect fixed_point_rect; |
+ if (!hw_plane->is_dummy()) { |
+ 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())); |
+ } |
+ |
+ plane_list->plane_list.push_back(hw_plane); |
+ hw_plane->set_owning_crtc(crtc_id); |
+ if (SetPlaneData(plane_list, hw_plane, plane, crtc_id, fixed_point_rect, |
+ crtc)) { |
+ hw_plane->set_in_use(true); |
+ } else { |
+ return false; |
+ } |
+ } |
+ return true; |
+} |
+ |
+void HardwareDisplayPlaneManager::ResetPlanes( |
+ HardwareDisplayPlaneList* plane_list, |
+ uint32_t crtc_id) { |
+ std::vector<HardwareDisplayPlane*> planes; |
+ planes.swap(plane_list->old_plane_list); |
+ for (auto* plane : planes) { |
+ if (plane->owning_crtc() == crtc_id) { |
+ plane->set_owning_crtc(0); |
+ plane->set_in_use(false); |
+ } else { |
+ plane_list->old_plane_list.push_back(plane); |
+ } |
+ } |
+} |
+ |
+} // namespace ui |