summaryrefslogtreecommitdiff
path: root/kms++/src/plane.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kms++/src/plane.cpp')
-rw-r--r--kms++/src/plane.cpp154
1 files changed, 154 insertions, 0 deletions
diff --git a/kms++/src/plane.cpp b/kms++/src/plane.cpp
new file mode 100644
index 0000000..3e17024
--- /dev/null
+++ b/kms++/src/plane.cpp
@@ -0,0 +1,154 @@
+#include <cstdio>
+#include <iostream>
+#include <unistd.h>
+#include <fcntl.h>
+#include <cassert>
+#include <xf86drm.h>
+#include <xf86drmMode.h>
+#include <algorithm>
+
+#include <kms++/kms++.h>
+
+using namespace std;
+
+namespace kms
+{
+struct PlanePriv {
+ drmModePlanePtr drm_plane;
+};
+
+Plane::Plane(Card& card, uint32_t id, uint32_t idx)
+ : DrmPropObject(card, id, DRM_MODE_OBJECT_PLANE, idx)
+{
+ m_priv = new PlanePriv();
+ m_priv->drm_plane = drmModeGetPlane(this->card().fd(), this->id());
+ assert(m_priv->drm_plane);
+}
+
+Plane::~Plane()
+{
+ drmModeFreePlane(m_priv->drm_plane);
+ delete m_priv;
+}
+
+bool Plane::supports_crtc(Crtc* crtc) const
+{
+ return m_priv->drm_plane->possible_crtcs & (1 << crtc->idx());
+}
+
+bool Plane::supports_format(PixelFormat fmt) const
+{
+ auto p = m_priv->drm_plane;
+
+ uint32_t fcc = pixel_format_to_fourcc(fmt);
+
+ for (unsigned i = 0; i < p->count_formats; ++i)
+ if (fcc == p->formats[i])
+ return true;
+
+ return false;
+}
+
+PlaneType Plane::plane_type() const
+{
+ if (card().has_universal_planes()) {
+ switch (get_prop_value("type")) {
+ case DRM_PLANE_TYPE_OVERLAY:
+ return PlaneType::Overlay;
+ case DRM_PLANE_TYPE_PRIMARY:
+ return PlaneType::Primary;
+ case DRM_PLANE_TYPE_CURSOR:
+ return PlaneType::Cursor;
+ default:
+ throw invalid_argument("Bad plane type");
+ }
+ } else {
+ return PlaneType::Overlay;
+ }
+}
+
+vector<Crtc*> Plane::get_possible_crtcs() const
+{
+ unsigned idx = 0;
+ vector<Crtc*> v;
+ auto crtcs = card().get_crtcs();
+
+ for (uint32_t crtc_mask = m_priv->drm_plane->possible_crtcs;
+ crtc_mask;
+ idx++, crtc_mask >>= 1) {
+ if ((crtc_mask & 1) == 0)
+ continue;
+
+ auto iter = find_if(crtcs.begin(), crtcs.end(), [idx](Crtc* crtc) { return crtc->idx() == idx; });
+
+ if (iter == crtcs.end())
+ continue;
+
+ v.push_back(*iter);
+ }
+
+ return v;
+}
+
+vector<uint32_t> Plane::get_fourccs() const
+{
+ auto p = m_priv->drm_plane;
+ vector<uint32_t> r;
+
+ for (unsigned i = 0; i < p->count_formats; ++i)
+ r.push_back(p->formats[i]);
+
+ return r;
+}
+
+vector<PixelFormat> Plane::get_formats() const
+{
+ auto p = m_priv->drm_plane;
+ vector<PixelFormat> r;
+
+ for (unsigned i = 0; i < p->count_formats; ++i)
+ try {
+ r.push_back(fourcc_to_pixel_format(p->formats[i]));
+ } catch (const std::invalid_argument&) {
+ // skip formats that are not supported
+ }
+
+ return r;
+}
+
+uint32_t Plane::crtc_id() const
+{
+ return m_priv->drm_plane->crtc_id;
+}
+
+uint32_t Plane::fb_id() const
+{
+ return m_priv->drm_plane->fb_id;
+}
+
+uint32_t Plane::crtc_x() const
+{
+ return m_priv->drm_plane->crtc_x;
+}
+
+uint32_t Plane::crtc_y() const
+{
+ return m_priv->drm_plane->crtc_y;
+}
+
+uint32_t Plane::x() const
+{
+ return m_priv->drm_plane->x;
+}
+
+uint32_t Plane::y() const
+{
+ return m_priv->drm_plane->y;
+}
+
+uint32_t Plane::gamma_size() const
+{
+ return m_priv->drm_plane->gamma_size;
+}
+
+} // namespace kms