mirror of
https://gitlab.freedesktop.org/emersion/libliftoff.git
synced 2025-01-13 20:01:35 +01:00
78cc612f10
Add a fallback case for finding an eligible CRTC for the selected connector. The encoder picked may not have been wired up to that connector by the previous DRM master.
148 lines
3.2 KiB
C
148 lines
3.2 KiB
C
#include <assert.h>
|
|
#include <drm_fourcc.h>
|
|
#include <stdio.h>
|
|
#include <stddef.h>
|
|
#include <sys/mman.h>
|
|
#include <xf86drm.h>
|
|
#include "common.h"
|
|
|
|
drmModeConnector *pick_connector(int drm_fd, drmModeRes *drm_res)
|
|
{
|
|
int i;
|
|
drmModeConnector *connector;
|
|
|
|
for (i = 0; i < drm_res->count_connectors; i++) {
|
|
connector = drmModeGetConnector(drm_fd, drm_res->connectors[i]);
|
|
if (connector->connection == DRM_MODE_CONNECTED) {
|
|
return connector;
|
|
}
|
|
drmModeFreeConnector(connector);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
drmModeCrtc *pick_crtc(int drm_fd, drmModeRes *drm_res,
|
|
drmModeConnector *connector)
|
|
{
|
|
drmModeEncoder *enc;
|
|
uint32_t crtc_id;
|
|
int i;
|
|
int j;
|
|
bool found;
|
|
|
|
enc = drmModeGetEncoder(drm_fd, connector->encoder_id);
|
|
|
|
if (enc) {
|
|
/* Current CRTC happens to be usable on the selected connector */
|
|
crtc_id = enc->crtc_id;
|
|
drmModeFreeEncoder(enc);
|
|
return drmModeGetCrtc(drm_fd, crtc_id);
|
|
} else {
|
|
/* Current CRTC used by this encoder can't drive the selected connector.
|
|
* Search all of them for a valid combination. */
|
|
for (i = 0, found = false; !found && i < connector->count_encoders; i++) {
|
|
enc = drmModeGetEncoder(drm_fd, connector->encoders[i]);
|
|
|
|
if (!enc) {
|
|
continue;
|
|
}
|
|
|
|
for (j = 0; !found && j < drm_res->count_crtcs; j++) {
|
|
/* Can the CRTC drive the connector? */
|
|
if (enc->possible_crtcs & (1 << j)) {
|
|
crtc_id = drm_res->crtcs[j];
|
|
found = true;
|
|
}
|
|
}
|
|
drmModeFreeEncoder(enc);
|
|
}
|
|
|
|
if (found) {
|
|
return drmModeGetCrtc(drm_fd, crtc_id);
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
void disable_all_crtcs_except(int drm_fd, drmModeRes *drm_res, uint32_t crtc_id)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < drm_res->count_crtcs; i++) {
|
|
if (drm_res->crtcs[i] == crtc_id) {
|
|
continue;
|
|
}
|
|
drmModeSetCrtc(drm_fd, drm_res->crtcs[i],
|
|
0, 0, 0, NULL, 0, NULL);
|
|
}
|
|
}
|
|
|
|
bool dumb_fb_init(struct dumb_fb *fb, int drm_fd, uint32_t format,
|
|
uint32_t width, uint32_t height)
|
|
{
|
|
int ret;
|
|
uint32_t fb_id;
|
|
|
|
assert(format == DRM_FORMAT_ARGB8888 || format == DRM_FORMAT_XRGB8888);
|
|
|
|
struct drm_mode_create_dumb create = {
|
|
.width = width,
|
|
.height = height,
|
|
.bpp = 32,
|
|
.flags = 0,
|
|
};
|
|
ret = drmIoctl(drm_fd, DRM_IOCTL_MODE_CREATE_DUMB, &create);
|
|
if (ret < 0) {
|
|
return false;
|
|
}
|
|
|
|
uint32_t handles[4] = { create.handle };
|
|
uint32_t strides[4] = { create.pitch };
|
|
uint32_t offsets[4] = { 0 };
|
|
ret = drmModeAddFB2(drm_fd, width, height, format, handles, strides,
|
|
offsets, &fb_id, 0);
|
|
if (ret < 0) {
|
|
return false;
|
|
}
|
|
|
|
fb->width = width;
|
|
fb->height = height;
|
|
fb->stride = create.pitch;
|
|
fb->size = create.size;
|
|
fb->handle = create.handle;
|
|
fb->id = fb_id;
|
|
return true;
|
|
}
|
|
|
|
void *dumb_fb_map(struct dumb_fb *fb, int drm_fd)
|
|
{
|
|
int ret;
|
|
|
|
struct drm_mode_map_dumb map = { .handle = fb->handle };
|
|
ret = drmIoctl(drm_fd, DRM_IOCTL_MODE_MAP_DUMB, &map);
|
|
if (ret < 0) {
|
|
return MAP_FAILED;
|
|
}
|
|
|
|
return mmap(0, fb->size, PROT_READ | PROT_WRITE, MAP_SHARED, drm_fd,
|
|
map.offset);
|
|
}
|
|
|
|
void dumb_fb_fill(struct dumb_fb *fb, int drm_fd, uint32_t color)
|
|
{
|
|
uint32_t *data;
|
|
size_t i;
|
|
|
|
data = dumb_fb_map(fb, drm_fd);
|
|
if (data == MAP_FAILED) {
|
|
return;
|
|
}
|
|
|
|
for (i = 0; i < fb->size / sizeof(uint32_t); i++) {
|
|
data[i] = color;
|
|
}
|
|
|
|
munmap(data, fb->size);
|
|
}
|