diff options
author | Rob Clark <robdclark@gmail.com> | 2017-02-22 11:36:12 -0500 |
---|---|---|
committer | Rob Clark <robdclark@gmail.com> | 2017-02-25 10:48:22 -0500 |
commit | 87e3ff5683ee54228b3e6e75f7d4de83901fadb0 (patch) | |
tree | 0aa767fb89bbf7c64670fb117f5a2b65cda9db4a /drm-atomic.c | |
parent | 1f1158fe1279e845420d3cb838eff3bbf2e5701a (diff) | |
download | kmscube-87e3ff5683ee54228b3e6e75f7d4de83901fadb0.tar.gz |
add atomic kms support
Based mostly on original atomic conversion from Gustavo Padovan.
Diffstat (limited to 'drm-atomic.c')
-rw-r--r-- | drm-atomic.c | 388 |
1 files changed, 388 insertions, 0 deletions
diff --git a/drm-atomic.c b/drm-atomic.c new file mode 100644 index 0000000..b4755e1 --- /dev/null +++ b/drm-atomic.c @@ -0,0 +1,388 @@ +/* + * Copyright (c) 2017 Rob Clark <rclark@redhat.com> + * + * Permission is hereby granted, free of charge, to any person obtaining a + * copy of this software and associated documentation files (the "Software"), + * to deal in the Software without restriction, including without limitation + * the rights to use, copy, modify, merge, publish, distribute, sub license, + * and/or sell copies of the Software, and to permit persons to whom the + * Software is furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice (including the + * next paragraph) shall be included in all copies or substantial portions + * of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER + * DEALINGS IN THE SOFTWARE. + */ + +#include <assert.h> +#include <errno.h> +#include <stdio.h> +#include <string.h> +#include <unistd.h> + +#include "common.h" +#include "drm.h" + +#define VOID2U64(x) ((uint64_t)(unsigned long)(x)) + +static struct drm drm; + + +static int add_connector_property(drmModeAtomicReq *req, uint32_t obj_id, + const char *name, uint64_t value) +{ + struct connector *obj = drm.connector; + unsigned int i; + int prop_id = 0; + + for (i = 0 ; i < obj->props->count_props ; i++) { + if (strcmp(obj->props_info[i]->name, name) == 0) { + prop_id = obj->props_info[i]->prop_id; + break; + } + } + + if (prop_id < 0) { + printf("no connector property: %s\n", name); + return -EINVAL; + } + + return drmModeAtomicAddProperty(req, obj_id, prop_id, value); +} + +static int add_crtc_property(drmModeAtomicReq *req, uint32_t obj_id, + const char *name, uint64_t value) +{ + struct crtc *obj = drm.crtc; + unsigned int i; + int prop_id = -1; + + for (i = 0 ; i < obj->props->count_props ; i++) { + if (strcmp(obj->props_info[i]->name, name) == 0) { + prop_id = obj->props_info[i]->prop_id; + break; + } + } + + if (prop_id < 0) { + printf("no crtc property: %s\n", name); + return -EINVAL; + } + + return drmModeAtomicAddProperty(req, obj_id, prop_id, value); +} + +static int add_plane_property(drmModeAtomicReq *req, uint32_t obj_id, + const char *name, uint64_t value) +{ + struct plane *obj = drm.plane; + unsigned int i; + int prop_id = -1; + + for (i = 0 ; i < obj->props->count_props ; i++) { + if (strcmp(obj->props_info[i]->name, name) == 0) { + prop_id = obj->props_info[i]->prop_id; + break; + } + } + + + if (prop_id < 0) { + printf("no plane property: %s\n", name); + return -EINVAL; + } + + return drmModeAtomicAddProperty(req, obj_id, prop_id, value); +} + +static int drm_atomic_commit(uint32_t fb_id, uint32_t flags) +{ + drmModeAtomicReq *req; + uint32_t plane_id = drm.plane->plane->plane_id; + uint32_t blob_id; + unsigned int i; + int ret; + + req = drmModeAtomicAlloc(); + + if (flags & DRM_MODE_ATOMIC_ALLOW_MODESET) { + if (add_connector_property(req, drm.connector_id, "CRTC_ID", + drm.crtc_id) < 0) + return -1; + + if (drmModeCreatePropertyBlob(drm.fd, drm.mode, sizeof(*drm.mode), + &blob_id) != 0) + return -1; + + if (add_crtc_property(req, drm.crtc_id, "MODE_ID", blob_id) < 0) + return -1; + + if (add_crtc_property(req, drm.crtc_id, "ACTIVE", 1) < 0) + return -1; + } + + add_plane_property(req, plane_id, "FB_ID", fb_id); + add_plane_property(req, plane_id, "CRTC_ID", drm.crtc_id); + add_plane_property(req, plane_id, "SRC_X", 0); + add_plane_property(req, plane_id, "SRC_Y", 0); + add_plane_property(req, plane_id, "SRC_W", drm.mode->hdisplay << 16); + add_plane_property(req, plane_id, "SRC_H", drm.mode->vdisplay << 16); + add_plane_property(req, plane_id, "CRTC_X", 0); + add_plane_property(req, plane_id, "CRTC_Y", 0); + add_plane_property(req, plane_id, "CRTC_W", drm.mode->hdisplay); + add_plane_property(req, plane_id, "CRTC_H", drm.mode->vdisplay); + + if (drm.kms_in_fence_fd != -1) { + add_crtc_property(req, drm.crtc_id, "OUT_FENCE_PTR", + VOID2U64(&drm.kms_out_fence_fd)); + add_plane_property(req, plane_id, "IN_FENCE_FD", drm.kms_in_fence_fd); + } + + ret = drmModeAtomicCommit(drm.fd, req, flags, NULL); + if (ret) + goto out; + + if (drm.kms_in_fence_fd != -1) { + close(drm.kms_in_fence_fd); + drm.kms_in_fence_fd = -1; + } + +out: + drmModeAtomicFree(req); + + return ret; +} + +static EGLSyncKHR create_fence(const struct egl *egl, int fd) +{ + EGLint attrib_list[] = { + EGL_SYNC_NATIVE_FENCE_FD_ANDROID, fd, + EGL_NONE, + }; + EGLSyncKHR fence = egl->eglCreateSyncKHR(egl->display, + EGL_SYNC_NATIVE_FENCE_ANDROID, attrib_list); + assert(fence); + return fence; +} + +static int atomic_run(const struct gbm *gbm, const struct egl *egl) +{ + struct gbm_bo *bo; + struct drm_fb *fb; + uint32_t i = 0; + int ret; + + if (!egl->eglDupNativeFenceFDANDROID) { + printf("no eglDupNativeFenceFDANDROID\n"); + return -1; + } + + eglSwapBuffers(egl->display, egl->surface); + bo = gbm_surface_lock_front_buffer(gbm->surface); + fb = drm_fb_get_from_bo(bo); + + drm.kms_in_fence_fd = -1; + + /* set mode: */ + ret = drm_atomic_commit(fb->fb_id, DRM_MODE_ATOMIC_ALLOW_MODESET); + if (ret) { + printf("failed to commit modeset: %s\n", strerror(errno)); + return ret; + } + + while (1) { + struct gbm_bo *next_bo; + EGLSyncKHR gpu_fence = NULL; /* out-fence from gpu, in-fence to kms */ + EGLSyncKHR kms_fence = NULL; /* in-fence to gpu, out-fence from kms */ + int gpu_fence_fd, kms_fence_fd; /* just for debugging */ + + kms_fence_fd = drm.kms_out_fence_fd; + + if (drm.kms_out_fence_fd != -1) { + kms_fence = create_fence(egl, drm.kms_out_fence_fd); + + /* driver now has ownership of the fence fd: */ + drm.kms_out_fence_fd = -1; + + /* wait "on the gpu" (ie. this won't necessarily block, but + * will block the rendering until fence is signaled), until + * the previous pageflip completes so we don't render into + * the buffer that is still on screen. + */ + egl->eglWaitSyncKHR(egl->display, kms_fence, 0); + egl->eglDestroySyncKHR(egl->display, kms_fence); + } + + egl->draw(i++); + + /* insert fence to be singled in cmdstream.. this fence will be + * signaled when gpu rendering done + */ + gpu_fence = create_fence(egl, EGL_NO_NATIVE_FENCE_FD_ANDROID); + + eglSwapBuffers(egl->display, egl->surface); + + /* after swapbuffers, gpu_fence should be flushed, so safe + * to get fd: + */ + drm.kms_in_fence_fd = egl->eglDupNativeFenceFDANDROID(egl->display, gpu_fence); + gpu_fence_fd = drm.kms_in_fence_fd; + egl->eglDestroySyncKHR(egl->display, gpu_fence); + assert(drm.kms_in_fence_fd != -1); + + next_bo = gbm_surface_lock_front_buffer(gbm->surface); + fb = drm_fb_get_from_bo(next_bo); + + /* + * Here you could also update drm plane layers if you want + * hw composition + */ + ret = drm_atomic_commit(fb->fb_id, DRM_MODE_ATOMIC_NONBLOCK); + if (ret) { + printf("failed to commit: %s\n", strerror(errno)); + return -1; + } + + /* release last buffer to render on again: */ + gbm_surface_release_buffer(gbm->surface, bo); + bo = next_bo; + } + + return ret; +} + +/* Pick a plane.. something that at a minimum can be connected to + * the chosen crtc, but prefer primary plane. + * + * Seems like there is some room for a drmModeObjectGetNamedProperty() + * type helper in libdrm.. + */ +static int get_plane_id(void) +{ + drmModePlaneResPtr plane_resources; + uint32_t i, j; + int ret = -EINVAL; + int found_primary = 0; + + plane_resources = drmModeGetPlaneResources(drm.fd); + if (!plane_resources) { + printf("drmModeGetPlaneResources failed: %s\n", strerror(errno)); + return -1; + } + + for (i = 0; (i < plane_resources->count_planes) && !found_primary; i++) { + uint32_t id = plane_resources->planes[i]; + drmModePlanePtr plane = drmModeGetPlane(drm.fd, id); + if (!plane) { + printf("drmModeGetPlane(%u) failed: %s\n", id, strerror(errno)); + continue; + } + + if (plane->possible_crtcs & (1 << drm.crtc_index)) { + drmModeObjectPropertiesPtr props = + drmModeObjectGetProperties(drm.fd, id, DRM_MODE_OBJECT_PLANE); + + /* primary or not, this plane is good enough to use: */ + ret = id; + + for (j = 0; j < props->count_props; j++) { + drmModePropertyPtr p = + drmModeGetProperty(drm.fd, props->props[j]); + + if ((strcmp(p->name, "type") == 0) && + (props->prop_values[j] == DRM_PLANE_TYPE_PRIMARY)) { + /* found our primary plane, lets use that: */ + found_primary = 1; + } + + drmModeFreeProperty(p); + } + + drmModeFreeObjectProperties(props); + } + + drmModeFreePlane(plane); + } + + drmModeFreePlaneResources(plane_resources); + + return ret; +} + +const struct drm * init_drm_atomic(const char *device) +{ + uint32_t plane_id; + int i, ret; + + ret = init_drm(&drm, device); + if (ret) + return NULL; + + ret = drmSetClientCap(drm.fd, DRM_CLIENT_CAP_ATOMIC, 1); + if (ret) { + printf("no atomic modesetting support: %s\n", strerror(errno)); + return NULL; + } + + ret = get_plane_id(); + if (!ret) { + printf("could not find a suitable plane\n"); + return NULL; + } else { + plane_id = ret; + } + + /* We only do single plane to single crtc to single connector, no + * fancy multi-monitor or multi-plane stuff. So just grab the + * plane/crtc/connector property info for one of each: + */ + drm.plane = calloc(1, sizeof(*drm.plane)); + drm.crtc = calloc(1, sizeof(*drm.crtc)); + drm.connector = calloc(1, sizeof(*drm.connector)); + +#define get_resource(type, Type, id) do { \ + drm.type->type = drmModeGet##Type(drm.fd, id); \ + if (!drm.type->type) { \ + printf("could not get %s %i: %s\n", \ + #type, id, strerror(errno)); \ + return NULL; \ + } \ + } while (0) + + get_resource(plane, Plane, plane_id); + get_resource(crtc, Crtc, drm.crtc_id); + get_resource(connector, Connector, drm.connector_id); + +#define get_properties(type, TYPE, id) do { \ + int i; \ + drm.type->props = drmModeObjectGetProperties(drm.fd, \ + id, DRM_MODE_OBJECT_##TYPE); \ + if (!drm.type->props) { \ + printf("could not get %s %u properties: %s\n", \ + #type, id, strerror(errno)); \ + return NULL; \ + } \ + drm.type->props_info = calloc(drm.type->props->count_props, \ + sizeof(drm.type->props_info)); \ + for (i = 0; i < drm.type->props->count_props; i++) { \ + drm.type->props_info[i] = drmModeGetProperty(drm.fd,\ + drm.type->props->props[i]); \ + } \ + } while (0) + + + get_properties(plane, PLANE, plane_id); + get_properties(crtc, CRTC, drm.crtc_id); + get_properties(connector, CONNECTOR, drm.connector_id); + + drm.run = atomic_run; + + return &drm; +} |