mirror of
https://gitlab.freedesktop.org/emersion/libliftoff.git
synced 2024-12-25 21:59:11 +01:00
8a971695bd
This allows to mimick i915's requirement of failing all commits which don't have the primary plane enabled. We could expand our test matrix to test with the param *and* without it, but since enabling the param just makes the atomic check more strict there's no real benefit of doing that. Enabling the param can only make tests fail more often, not the other way around.
502 lines
10 KiB
C
502 lines
10 KiB
C
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <inttypes.h>
|
|
#include <stdbool.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <sys/stat.h>
|
|
#include <unistd.h>
|
|
#include "libdrm_mock.h"
|
|
|
|
#define MAX_PLANES 64
|
|
#define MAX_LAYERS 512
|
|
#define MAX_PLANE_PROPS 64
|
|
#define MAX_REQ_PROPS 1024
|
|
|
|
uint32_t liftoff_mock_drm_crtc_id = 0xCC000000;
|
|
size_t liftoff_mock_commit_count = 0;
|
|
bool liftoff_mock_require_primary_plane = false;
|
|
|
|
struct liftoff_mock_plane {
|
|
uint32_t id;
|
|
struct liftoff_layer *compatible_layers[MAX_LAYERS];
|
|
bool enabled_props[MAX_PLANE_PROPS];
|
|
uint64_t prop_values[MAX_PLANE_PROPS];
|
|
};
|
|
|
|
struct liftoff_mock_prop {
|
|
uint32_t obj_id, prop_id;
|
|
uint64_t value;
|
|
};
|
|
|
|
struct _drmModeAtomicReq {
|
|
struct liftoff_mock_prop props[MAX_REQ_PROPS];
|
|
int cursor;
|
|
};
|
|
|
|
static int mock_pipe[2] = {-1, -1};
|
|
static struct liftoff_mock_plane mock_planes[MAX_PLANES];
|
|
static struct liftoff_layer *mock_fbs[MAX_LAYERS];
|
|
|
|
enum plane_prop {
|
|
PLANE_TYPE,
|
|
PLANE_FB_ID,
|
|
PLANE_CRTC_ID,
|
|
};
|
|
|
|
static const char *basic_plane_props[] = {
|
|
[PLANE_TYPE] = "type",
|
|
[PLANE_FB_ID] = "FB_ID",
|
|
[PLANE_CRTC_ID] = "CRTC_ID",
|
|
"CRTC_X",
|
|
"CRTC_Y",
|
|
"CRTC_W",
|
|
"CRTC_H",
|
|
"SRC_X",
|
|
"SRC_Y",
|
|
"SRC_W",
|
|
"SRC_H",
|
|
};
|
|
|
|
static const size_t basic_plane_props_len = sizeof(basic_plane_props) /
|
|
sizeof(basic_plane_props[0]);
|
|
|
|
static drmModePropertyRes plane_props[MAX_PLANE_PROPS] = {0};
|
|
|
|
static size_t plane_props_len = 0;
|
|
|
|
static void assert_drm_fd(int fd)
|
|
{
|
|
int ret;
|
|
struct stat stat_got, stat_want;
|
|
|
|
ret = fstat(mock_pipe[0], &stat_want);
|
|
assert(ret == 0);
|
|
ret = fstat(fd, &stat_got);
|
|
assert(ret == 0);
|
|
|
|
assert(stat_got.st_dev == stat_want.st_dev &&
|
|
stat_got.st_ino == stat_want.st_ino);
|
|
}
|
|
|
|
static uint32_t register_prop(const drmModePropertyRes *prop)
|
|
{
|
|
drmModePropertyRes *dst;
|
|
|
|
assert(plane_props_len < MAX_PLANE_PROPS);
|
|
dst = &plane_props[plane_props_len];
|
|
memcpy(dst, prop, sizeof(*dst));
|
|
dst->prop_id = 0xB0000000 + plane_props_len;
|
|
plane_props_len++;
|
|
|
|
return dst->prop_id;
|
|
}
|
|
|
|
static void init_basic_props(void)
|
|
{
|
|
size_t i;
|
|
|
|
if (plane_props_len > 0)
|
|
return;
|
|
|
|
for (i = 0; i < basic_plane_props_len; i++) {
|
|
drmModePropertyRes prop = {0};
|
|
strncpy(prop.name, basic_plane_props[i], sizeof(prop.name) - 1);
|
|
/* TODO: fill flags */
|
|
register_prop(&prop);
|
|
}
|
|
}
|
|
|
|
int liftoff_mock_drm_open(void)
|
|
{
|
|
int ret;
|
|
|
|
assert(mock_pipe[0] < 0);
|
|
ret = pipe(mock_pipe);
|
|
assert(ret == 0);
|
|
|
|
init_basic_props();
|
|
|
|
return mock_pipe[0];
|
|
}
|
|
|
|
struct liftoff_mock_plane *liftoff_mock_drm_create_plane(int type)
|
|
{
|
|
struct liftoff_mock_plane *plane;
|
|
size_t i;
|
|
|
|
assert(mock_pipe[0] < 0);
|
|
|
|
init_basic_props();
|
|
|
|
i = 0;
|
|
plane = &mock_planes[0];
|
|
while (plane->id != 0) {
|
|
plane++;
|
|
i++;
|
|
}
|
|
|
|
plane->id = 0xEE000000 + i;
|
|
plane->prop_values[PLANE_TYPE] = type;
|
|
|
|
for (size_t i = 0; i < basic_plane_props_len; i++) {
|
|
plane->enabled_props[i] = true;
|
|
}
|
|
|
|
return plane;
|
|
}
|
|
|
|
struct liftoff_mock_plane *liftoff_mock_drm_get_plane(uint32_t id)
|
|
{
|
|
struct liftoff_mock_plane *plane;
|
|
|
|
plane = &mock_planes[0];
|
|
while (plane->id != 0) {
|
|
if (plane->id == id) {
|
|
return plane;
|
|
}
|
|
plane++;
|
|
}
|
|
|
|
abort(); // unreachable
|
|
}
|
|
|
|
void liftoff_mock_plane_add_compatible_layer(struct liftoff_mock_plane *plane,
|
|
struct liftoff_layer *layer)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < MAX_LAYERS; i++) {
|
|
if (plane->compatible_layers[i] == NULL) {
|
|
plane->compatible_layers[i] = layer;
|
|
return;
|
|
}
|
|
}
|
|
|
|
abort(); // unreachable
|
|
}
|
|
|
|
uint32_t liftoff_mock_drm_create_fb(struct liftoff_layer *layer)
|
|
{
|
|
size_t i;
|
|
|
|
i = 0;
|
|
while (mock_fbs[i] != 0) {
|
|
i++;
|
|
}
|
|
|
|
mock_fbs[i] = layer;
|
|
|
|
return 0xFB000000 + i;
|
|
}
|
|
|
|
static bool mock_atomic_req_get_property(drmModeAtomicReq *req, uint32_t obj_id,
|
|
enum plane_prop prop, uint64_t *value)
|
|
{
|
|
ssize_t i;
|
|
uint32_t prop_id;
|
|
|
|
prop_id = 0xB0000000 + prop;
|
|
for (i = req->cursor - 1; i >= 0; i--) {
|
|
if (req->props[i].obj_id == obj_id &&
|
|
req->props[i].prop_id == prop_id) {
|
|
*value = req->props[i].value;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
static struct liftoff_layer *mock_fb_get_layer(uint32_t fb_id)
|
|
{
|
|
size_t i;
|
|
|
|
if (fb_id == 0) {
|
|
return NULL;
|
|
}
|
|
|
|
assert((fb_id & 0xFF000000) == 0xFB000000);
|
|
|
|
i = fb_id & 0x00FFFFFF;
|
|
assert(i < MAX_LAYERS);
|
|
|
|
return mock_fbs[i];
|
|
}
|
|
|
|
struct liftoff_layer *liftoff_mock_plane_get_layer(struct liftoff_mock_plane *plane)
|
|
{
|
|
return mock_fb_get_layer(plane->prop_values[PLANE_FB_ID]);
|
|
}
|
|
|
|
static size_t get_prop_index(uint32_t id)
|
|
{
|
|
size_t i;
|
|
|
|
assert((id & 0xFF000000) == 0xB0000000);
|
|
|
|
i = id & 0x00FFFFFF;
|
|
assert(i < plane_props_len);
|
|
|
|
return i;
|
|
}
|
|
|
|
uint32_t liftoff_mock_plane_add_property(struct liftoff_mock_plane *plane,
|
|
const drmModePropertyRes *prop)
|
|
{
|
|
uint32_t prop_id;
|
|
|
|
prop_id = register_prop(prop);
|
|
plane->enabled_props[get_prop_index(prop_id)] = true;
|
|
if (prop->count_values == 1) {
|
|
plane->prop_values[get_prop_index(prop_id)] = prop->values[0];
|
|
}
|
|
return prop_id;
|
|
}
|
|
|
|
static void apply_atomic_req(drmModeAtomicReq *req)
|
|
{
|
|
int i;
|
|
size_t prop_index;
|
|
struct liftoff_mock_prop *prop;
|
|
struct liftoff_mock_plane *plane;
|
|
|
|
for (i = 0; i < req->cursor; i++) {
|
|
prop = &req->props[i];
|
|
plane = liftoff_mock_drm_get_plane(prop->obj_id);
|
|
prop_index = get_prop_index(prop->prop_id);
|
|
plane->prop_values[prop_index] = prop->value;
|
|
fprintf(stderr, "libdrm_mock: plane %"PRIu32": "
|
|
"setting %s = %"PRIu64"\n", plane->id,
|
|
plane_props[prop_index].name, prop->value);
|
|
}
|
|
}
|
|
|
|
int drmModeAtomicCommit(int fd, drmModeAtomicReq *req, uint32_t flags,
|
|
void *user_data)
|
|
{
|
|
size_t i, j;
|
|
struct liftoff_mock_plane *plane;
|
|
uint64_t type, fb_id, crtc_id;
|
|
bool has_fb, has_crtc, found;
|
|
bool any_plane_enabled, primary_plane_enabled;
|
|
struct liftoff_layer *layer;
|
|
|
|
assert_drm_fd(fd);
|
|
assert(flags == DRM_MODE_ATOMIC_TEST_ONLY || flags == 0);
|
|
|
|
liftoff_mock_commit_count++;
|
|
|
|
any_plane_enabled = false;
|
|
primary_plane_enabled = false;
|
|
for (i = 0; i < MAX_PLANES; i++) {
|
|
plane = &mock_planes[i];
|
|
if (plane->id == 0) {
|
|
break;
|
|
}
|
|
|
|
type = plane->prop_values[PLANE_TYPE];
|
|
fb_id = plane->prop_values[PLANE_FB_ID];
|
|
crtc_id = plane->prop_values[PLANE_CRTC_ID];
|
|
mock_atomic_req_get_property(req, plane->id, PLANE_FB_ID,
|
|
&fb_id);
|
|
mock_atomic_req_get_property(req, plane->id, PLANE_CRTC_ID,
|
|
&crtc_id);
|
|
|
|
has_fb = fb_id != 0;
|
|
has_crtc = crtc_id != 0;
|
|
|
|
if (has_fb != has_crtc) {
|
|
fprintf(stderr, "libdrm_mock: plane %u: both FB_ID and "
|
|
"CRTC_ID must be set or unset together "
|
|
"(FB_ID = %"PRIu64", CRTC_ID = %"PRIu64")\n",
|
|
plane->id, fb_id, crtc_id);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (has_fb) {
|
|
if (crtc_id != liftoff_mock_drm_crtc_id) {
|
|
fprintf(stderr, "libdrm_mock: plane %u: "
|
|
"invalid CRTC_ID\n", plane->id);
|
|
return -EINVAL;
|
|
}
|
|
layer = mock_fb_get_layer(fb_id);
|
|
if (layer == NULL) {
|
|
fprintf(stderr, "libdrm_mock: plane %u: "
|
|
"invalid FB_ID\n", plane->id);
|
|
return -EINVAL;
|
|
}
|
|
found = false;
|
|
for (j = 0; j < MAX_LAYERS; j++) {
|
|
if (plane->compatible_layers[j] == layer) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!found) {
|
|
fprintf(stderr, "libdrm_mock: plane %u: "
|
|
"layer %p is not compatible\n",
|
|
plane->id, (void *)layer);
|
|
return -EINVAL;
|
|
}
|
|
|
|
any_plane_enabled = true;
|
|
if (type == DRM_PLANE_TYPE_PRIMARY) {
|
|
primary_plane_enabled = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (liftoff_mock_require_primary_plane && any_plane_enabled &&
|
|
!primary_plane_enabled) {
|
|
fprintf(stderr, "libdrm_mock: cannot light up CRTC without "
|
|
"enabling the primary plane\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!(flags & DRM_MODE_ATOMIC_TEST_ONLY)) {
|
|
apply_atomic_req(req);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
drmModeRes *drmModeGetResources(int fd)
|
|
{
|
|
drmModeRes *res;
|
|
|
|
assert_drm_fd(fd);
|
|
|
|
res = calloc(1, sizeof(*res));
|
|
res->count_crtcs = 1;
|
|
res->crtcs = &liftoff_mock_drm_crtc_id;
|
|
return res;
|
|
}
|
|
|
|
void drmModeFreeResources(drmModeRes *res)
|
|
{
|
|
free(res);
|
|
}
|
|
|
|
drmModePlaneRes *drmModeGetPlaneResources(int fd)
|
|
{
|
|
static uint32_t plane_ids[MAX_PLANES];
|
|
drmModePlaneRes *res;
|
|
size_t i;
|
|
|
|
assert_drm_fd(fd);
|
|
|
|
for (i = 0; i < MAX_PLANES; i++) {
|
|
if (mock_planes[i].id == 0) {
|
|
break;
|
|
}
|
|
plane_ids[i] = mock_planes[i].id;
|
|
}
|
|
|
|
res = calloc(1, sizeof(*res));
|
|
res->count_planes = i;
|
|
res->planes = plane_ids;
|
|
return res;
|
|
}
|
|
|
|
void drmModeFreePlaneResources(drmModePlaneRes *res)
|
|
{
|
|
free(res);
|
|
}
|
|
|
|
drmModePlane *drmModeGetPlane(int fd, uint32_t id)
|
|
{
|
|
drmModePlane *plane;
|
|
|
|
assert_drm_fd(fd);
|
|
|
|
plane = calloc(1, sizeof(*plane));
|
|
plane->plane_id = id;
|
|
plane->possible_crtcs = 1 << 0;
|
|
return plane;
|
|
}
|
|
|
|
void drmModeFreePlane(drmModePlane *plane) {
|
|
free(plane);
|
|
}
|
|
|
|
drmModeObjectProperties *drmModeObjectGetProperties(int fd, uint32_t obj_id,
|
|
uint32_t obj_type)
|
|
{
|
|
struct liftoff_mock_plane *plane;
|
|
drmModeObjectProperties *props;
|
|
size_t i;
|
|
|
|
assert_drm_fd(fd);
|
|
assert(obj_type == DRM_MODE_OBJECT_PLANE);
|
|
|
|
plane = NULL;
|
|
for (i = 0; i < MAX_PLANES; i++) {
|
|
if (mock_planes[i].id == obj_id) {
|
|
plane = &mock_planes[i];
|
|
break;
|
|
}
|
|
}
|
|
assert(plane != NULL);
|
|
|
|
props = calloc(1, sizeof(*props));
|
|
props->props = calloc(plane_props_len, sizeof(uint32_t));
|
|
props->prop_values = calloc(plane_props_len, sizeof(uint64_t));
|
|
for (i = 0; i < plane_props_len; i++) {
|
|
if (!plane->enabled_props[i])
|
|
continue;
|
|
props->props[props->count_props] = plane_props[i].prop_id;
|
|
props->prop_values[props->count_props] = plane->prop_values[i];
|
|
props->count_props++;
|
|
}
|
|
return props;
|
|
}
|
|
|
|
void drmModeFreeObjectProperties(drmModeObjectProperties *props) {
|
|
free(props->props);
|
|
free(props->prop_values);
|
|
free(props);
|
|
}
|
|
|
|
drmModePropertyRes *drmModeGetProperty(int fd, uint32_t id)
|
|
{
|
|
assert_drm_fd(fd);
|
|
|
|
return &plane_props[get_prop_index(id)];
|
|
}
|
|
|
|
void drmModeFreeProperty(drmModePropertyRes *prop) {
|
|
/* Owned by plane_props */
|
|
}
|
|
|
|
drmModeAtomicReq *drmModeAtomicAlloc(void)
|
|
{
|
|
return calloc(1, sizeof(drmModeAtomicReq));
|
|
}
|
|
|
|
void drmModeAtomicFree(drmModeAtomicReq *req)
|
|
{
|
|
free(req);
|
|
}
|
|
|
|
int drmModeAtomicAddProperty(drmModeAtomicReq *req, uint32_t obj_id,
|
|
uint32_t prop_id, uint64_t value)
|
|
{
|
|
assert((size_t)req->cursor < sizeof(req->props) / sizeof(req->props[0]));
|
|
req->props[req->cursor].obj_id = obj_id;
|
|
req->props[req->cursor].prop_id = prop_id;
|
|
req->props[req->cursor].value = value;
|
|
req->cursor++;
|
|
return req->cursor;
|
|
}
|
|
|
|
int drmModeAtomicGetCursor(drmModeAtomicReq *req)
|
|
{
|
|
return req->cursor;
|
|
}
|
|
|
|
void drmModeAtomicSetCursor(drmModeAtomicReq *req, int cursor)
|
|
{
|
|
req->cursor = cursor;
|
|
}
|