aboutsummaryrefslogtreecommitdiff
path: root/backend/drm/drm.c
diff options
context:
space:
mode:
authorScott Anderson <ascent12@hotmail.com>2017-07-20 20:51:59 +1200
committerScott Anderson <ascent12@hotmail.com>2017-08-06 15:49:02 +1200
commit7b772e1a4b8c7a06fbcf0ec57766af3a5384bf0c (patch)
treeae172598a5a8b8e20a61054475f0a16b02f28fa1 /backend/drm/drm.c
parent601bbad5d83387336028c5a721e8e85fccbbf844 (diff)
DRM resource initalisation
Diffstat (limited to 'backend/drm/drm.c')
-rw-r--r--backend/drm/drm.c133
1 files changed, 132 insertions, 1 deletions
diff --git a/backend/drm/drm.c b/backend/drm/drm.c
index 6e969788..7c27c046 100644
--- a/backend/drm/drm.c
+++ b/backend/drm/drm.c
@@ -15,7 +15,7 @@
#include <wlr/backend/interface.h>
#include <wlr/interfaces/wlr_output.h>
#include <wlr/util/log.h>
-#include "backend/drm.h"
+#include "drm.h"
static const char *conn_name[] = {
[DRM_MODE_CONNECTOR_Unknown] = "Unknown",
@@ -37,6 +37,137 @@ static const char *conn_name[] = {
[DRM_MODE_CONNECTOR_DSI] = "DSI",
};
+static int cmp_plane(const void *arg1, const void *arg2)
+{
+ const struct wlr_drm_plane *a = arg1;
+ const struct wlr_drm_plane *b = arg2;
+
+ return (int)a->type - (int)b->type;
+}
+
+static bool init_planes(struct wlr_backend_state *drm)
+{
+ drmModePlaneRes *plane_res = drmModeGetPlaneResources(drm->fd);
+ if (!plane_res) {
+ wlr_log_errno(L_ERROR, "Failed to get DRM plane resources");
+ return false;
+ }
+
+ wlr_log(L_INFO, "Found %"PRIu32" DRM planes", plane_res->count_planes);
+
+ if (plane_res->count_planes == 0) {
+ drmModeFreePlaneResources(plane_res);
+ return true;
+ }
+
+ size_t num_planes = plane_res->count_planes;
+ struct wlr_drm_plane *planes = calloc(num_planes, sizeof(*planes));
+ if (!planes) {
+ wlr_log_errno(L_ERROR, "Allocation failed");
+ goto error_res;
+ }
+
+ size_t num_overlay = 0;
+ size_t num_primary = 0;
+ size_t num_cursor = 0;
+
+ for (size_t i = 0; i < num_planes; ++i) {
+ struct wlr_drm_plane *p = &planes[i];
+
+ drmModePlane *plane = drmModeGetPlane(drm->fd, plane_res->planes[i]);
+ if (!plane) {
+ wlr_log_errno(L_ERROR, "Failed to get DRM plane");
+ goto error_planes;
+ }
+
+ p->id = plane->plane_id;
+ p->possible_crtcs = plane->possible_crtcs;
+ uint64_t type;
+
+ if (!wlr_drm_get_plane_props(drm->fd, p->id, &p->props) ||
+ !wlr_drm_get_prop(drm->fd, p->id, p->props.type, &type)) {
+ drmModeFreePlane(plane);
+ goto error_planes;
+ }
+
+ p->type = type;
+
+ switch (type) {
+ case DRM_PLANE_TYPE_OVERLAY:
+ ++num_overlay;
+ break;
+ case DRM_PLANE_TYPE_PRIMARY:
+ ++num_primary;
+ break;
+ case DRM_PLANE_TYPE_CURSOR:
+ ++num_cursor;
+ break;
+ }
+
+ drmModeFreePlane(plane);
+ }
+
+ wlr_log(L_INFO, "(%zu overlay, %zu primary, %zu cursor)",
+ num_overlay, num_primary, num_cursor);
+
+ qsort(planes, num_planes, sizeof(*planes), cmp_plane);
+
+ drm->num_planes = num_planes;
+ drm->num_overlay_planes = num_overlay;
+ drm->num_primary_planes = num_primary;
+ drm->num_cursor_planes = num_cursor;
+
+ drm->planes = planes;
+ drm->overlay_planes = planes;
+ drm->primary_planes = planes + num_overlay;
+ drm->cursor_planes = planes + num_overlay + num_primary;
+
+ return true;
+
+error_planes:
+ free(planes);
+error_res:
+ drmModeFreePlaneResources(plane_res);
+ return false;
+}
+
+bool wlr_drm_init_resources(struct wlr_backend_state *drm) {
+ drmModeRes *res = drmModeGetResources(drm->fd);
+ if (!res) {
+ wlr_log_errno(L_ERROR, "Failed to get DRM resources");
+ return false;
+ }
+
+ wlr_log(L_INFO, "Found %d DRM CRTCs", res->count_crtcs);
+
+ drm->num_crtcs = res->count_crtcs;
+ drm->crtcs = calloc(drm->num_crtcs, sizeof(drm->crtcs[0]));
+ if (!drm->crtcs) {
+ wlr_log_errno(L_ERROR, "Allocation failed");
+ goto error_res;
+ }
+
+ for (size_t i = 0; i < drm->num_crtcs; ++i) {
+ struct wlr_drm_crtc *crtc = &drm->crtcs[i];
+ crtc->id = res->crtcs[i];
+ wlr_drm_get_crtc_props(drm->fd, crtc->id, &crtc->props);
+ }
+
+ if (!init_planes(drm)) {
+ goto error_crtcs;
+ }
+
+ drmModeFreeResources(res);
+
+ return true;
+
+error_crtcs:
+ free(drm->crtcs);
+error_res:
+ drmModeFreeResources(res);
+ return false;
+}
+
bool wlr_drm_renderer_init(struct wlr_drm_renderer *renderer, int fd) {
renderer->gbm = gbm_create_device(fd);
if (!renderer->gbm) {