libliftoff/device.c

117 lines
2.7 KiB
C
Raw Normal View History

2020-01-13 12:22:47 +01:00
#include <errno.h>
#include <inttypes.h>
2019-08-21 22:07:37 +02:00
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "log.h"
2019-08-21 22:07:37 +02:00
#include "private.h"
struct liftoff_device *liftoff_device_create(int drm_fd)
2019-08-21 22:07:37 +02:00
{
struct liftoff_device *device;
drmModeRes *drm_res;
2019-08-21 22:07:37 +02:00
drmModePlaneRes *drm_plane_res;
uint32_t i;
device = calloc(1, sizeof(*device));
if (device == NULL) {
liftoff_log_errno(LIFTOFF_ERROR, "calloc");
2019-08-21 22:07:37 +02:00
return NULL;
}
liftoff_log(LIFTOFF_DEBUG, "\nCreating device for fd %d. ", drm_fd);
liftoff_list_init(&device->planes);
liftoff_list_init(&device->outputs);
device->drm_fd = dup(drm_fd);
if (device->drm_fd < 0) {
liftoff_log_errno(LIFTOFF_ERROR, "dup");
liftoff_device_destroy(device);
2019-08-21 22:07:37 +02:00
return NULL;
}
drm_res = drmModeGetResources(drm_fd);
if (drm_res == NULL) {
liftoff_log_errno(LIFTOFF_ERROR, "drmModeGetResources");
liftoff_device_destroy(device);
return NULL;
}
device->crtcs = malloc(drm_res->count_crtcs * sizeof(uint32_t));
if (device->crtcs == NULL) {
liftoff_log_errno(LIFTOFF_ERROR, "malloc");
drmModeFreeResources(drm_res);
liftoff_device_destroy(device);
return NULL;
}
device->crtcs_len = drm_res->count_crtcs;
memcpy(device->crtcs, drm_res->crtcs,
drm_res->count_crtcs * sizeof(uint32_t));
drmModeFreeResources(drm_res);
2019-08-21 22:07:37 +02:00
/* TODO: allow users to choose which layers to hand over */
drm_plane_res = drmModeGetPlaneResources(drm_fd);
if (drm_plane_res == NULL) {
liftoff_log_errno(LIFTOFF_ERROR, "drmModeGetPlaneResources");
liftoff_device_destroy(device);
2019-08-21 22:07:37 +02:00
return NULL;
}
liftoff_log(LIFTOFF_DEBUG, "The device has %"PRIu32 " planes:", drm_plane_res->count_planes);
2019-08-21 22:07:37 +02:00
for (i = 0; i < drm_plane_res->count_planes; i++) {
if (i < 9 && drm_plane_res->count_planes >= 9) {
debug_cnt(device, " ");
}
debug_cnt(device, "[%d] ", i + 1);
if (plane_create(device, drm_plane_res->planes[i]) == NULL) {
liftoff_device_destroy(device);
2019-08-21 22:07:37 +02:00
return NULL;
}
}
drmModeFreePlaneResources(drm_plane_res);
return device;
2019-08-21 22:07:37 +02:00
}
void liftoff_device_destroy(struct liftoff_device *device)
2019-08-21 22:07:37 +02:00
{
struct liftoff_plane *plane, *tmp;
2019-08-21 22:07:37 +02:00
if (device == NULL) {
return;
}
close(device->drm_fd);
liftoff_list_for_each_safe(plane, tmp, &device->planes, link) {
plane_destroy(plane);
2019-08-21 22:07:37 +02:00
}
free(device->crtcs);
free(device->log_buf);
free(device);
2019-08-21 22:07:37 +02:00
}
2020-01-13 12:22:47 +01:00
bool device_test_commit(struct liftoff_device *device,
drmModeAtomicReq *req, bool *compatible)
{
int ret;
do {
ret = drmModeAtomicCommit(device->drm_fd, req,
DRM_MODE_ATOMIC_TEST_ONLY, NULL);
} while (-ret == EINTR || -ret == EAGAIN);
2020-01-13 12:22:47 +01:00
if (ret == 0) {
*compatible = true;
} else if (-ret == EINVAL || -ret == ERANGE) {
*compatible = false;
} else {
liftoff_log_errno(LIFTOFF_ERROR, "drmModeAtomicCommit");
*compatible = false;
return false;
}
return true;
}