2019-09-25 14:06:46 +02:00
|
|
|
#include <assert.h>
|
2019-09-25 13:08:07 +02:00
|
|
|
#include <drm_fourcc.h>
|
2019-11-07 23:03:41 +01:00
|
|
|
#include <stdio.h>
|
2019-09-25 13:08:07 +02:00
|
|
|
#include <stddef.h>
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <xf86drm.h>
|
|
|
|
#include "common.h"
|
|
|
|
|
2021-08-13 22:02:33 +02:00
|
|
|
drmModeConnector *
|
|
|
|
pick_connector(int drm_fd, drmModeRes *drm_res)
|
2019-09-25 13:08:07 +02:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-08-13 22:02:33 +02:00
|
|
|
drmModeCrtc *
|
|
|
|
pick_crtc(int drm_fd, drmModeRes *drm_res, drmModeConnector *connector)
|
2019-09-25 13:08:07 +02:00
|
|
|
{
|
|
|
|
drmModeEncoder *enc;
|
|
|
|
uint32_t crtc_id;
|
2019-11-07 23:03:41 +01:00
|
|
|
int i;
|
|
|
|
int j;
|
|
|
|
bool found;
|
2019-09-25 13:08:07 +02:00
|
|
|
|
|
|
|
enc = drmModeGetEncoder(drm_fd, connector->encoder_id);
|
|
|
|
|
2019-11-07 23:03:41 +01:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
2019-09-25 13:08:07 +02:00
|
|
|
}
|
|
|
|
|
2021-08-13 22:02:33 +02:00
|
|
|
void
|
|
|
|
disable_all_crtcs_except(int drm_fd, drmModeRes *drm_res, uint32_t crtc_id)
|
2019-09-25 13:08:07 +02:00
|
|
|
{
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-13 22:02:33 +02:00
|
|
|
bool
|
|
|
|
dumb_fb_init(struct dumb_fb *fb, int drm_fd, uint32_t format, uint32_t width,
|
|
|
|
uint32_t height)
|
2019-09-25 13:08:07 +02:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
uint32_t fb_id;
|
2023-02-16 20:03:42 +01:00
|
|
|
struct drm_mode_create_dumb create;
|
|
|
|
uint32_t handles[4] = {0};
|
|
|
|
uint32_t strides[4] = {0};
|
|
|
|
uint32_t offsets[4] = { 0 };
|
2019-09-25 14:06:46 +02:00
|
|
|
|
|
|
|
assert(format == DRM_FORMAT_ARGB8888 || format == DRM_FORMAT_XRGB8888);
|
2019-09-25 13:08:07 +02:00
|
|
|
|
2023-02-16 20:03:42 +01:00
|
|
|
create = (struct drm_mode_create_dumb) {
|
2019-09-25 13:08:07 +02:00
|
|
|
.width = width,
|
|
|
|
.height = height,
|
|
|
|
.bpp = 32,
|
|
|
|
.flags = 0,
|
|
|
|
};
|
|
|
|
ret = drmIoctl(drm_fd, DRM_IOCTL_MODE_CREATE_DUMB, &create);
|
|
|
|
if (ret < 0) {
|
2019-09-25 14:06:46 +02:00
|
|
|
return false;
|
2019-09-25 13:08:07 +02:00
|
|
|
}
|
|
|
|
|
2023-02-16 20:03:42 +01:00
|
|
|
handles[0] = create.handle;
|
|
|
|
strides[0] = create.pitch;
|
2019-09-25 14:06:46 +02:00
|
|
|
ret = drmModeAddFB2(drm_fd, width, height, format, handles, strides,
|
2019-09-25 13:08:07 +02:00
|
|
|
offsets, &fb_id, 0);
|
|
|
|
if (ret < 0) {
|
2019-09-25 14:06:46 +02:00
|
|
|
return false;
|
2019-09-25 13:08:07 +02:00
|
|
|
}
|
|
|
|
|
2019-09-25 14:06:46 +02:00
|
|
|
fb->width = width;
|
|
|
|
fb->height = height;
|
|
|
|
fb->stride = create.pitch;
|
|
|
|
fb->size = create.size;
|
|
|
|
fb->handle = create.handle;
|
|
|
|
fb->id = fb_id;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-08-13 22:02:33 +02:00
|
|
|
void *
|
|
|
|
dumb_fb_map(struct dumb_fb *fb, int drm_fd)
|
2019-09-25 14:06:46 +02:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
struct drm_mode_map_dumb map = { .handle = fb->handle };
|
2019-09-25 13:08:07 +02:00
|
|
|
ret = drmIoctl(drm_fd, DRM_IOCTL_MODE_MAP_DUMB, &map);
|
|
|
|
if (ret < 0) {
|
2019-09-25 14:06:46 +02:00
|
|
|
return MAP_FAILED;
|
2019-09-25 13:08:07 +02:00
|
|
|
}
|
|
|
|
|
2019-09-25 14:06:46 +02:00
|
|
|
return mmap(0, fb->size, PROT_READ | PROT_WRITE, MAP_SHARED, drm_fd,
|
2019-09-25 13:08:07 +02:00
|
|
|
map.offset);
|
2019-09-25 14:06:46 +02:00
|
|
|
}
|
|
|
|
|
2021-08-13 22:02:33 +02:00
|
|
|
void
|
|
|
|
dumb_fb_fill(struct dumb_fb *fb, int drm_fd, uint32_t color)
|
2019-09-25 14:06:46 +02:00
|
|
|
{
|
|
|
|
uint32_t *data;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
data = dumb_fb_map(fb, drm_fd);
|
2019-09-25 13:08:07 +02:00
|
|
|
if (data == MAP_FAILED) {
|
2019-09-25 14:06:46 +02:00
|
|
|
return;
|
2019-09-25 13:08:07 +02:00
|
|
|
}
|
|
|
|
|
2019-09-25 14:06:46 +02:00
|
|
|
for (i = 0; i < fb->size / sizeof(uint32_t); i++) {
|
2019-09-25 13:08:07 +02:00
|
|
|
data[i] = color;
|
|
|
|
}
|
|
|
|
|
2019-09-25 14:06:46 +02:00
|
|
|
munmap(data, fb->size);
|
2019-09-25 13:08:07 +02:00
|
|
|
}
|