Cleaned some GBM code, looks great.

This commit is contained in:
2021-08-16 02:00:41 -04:00
parent 40a7041295
commit 0557ba3274

View File

@ -101,6 +101,49 @@ static uint32_t find_crtc_for_connector(
} }
static void drm_fb_destroy_callback(struct gbm_bo *bo, void *data) {
struct drm_fb *fb = data;
struct gbm_device *gbm = gbm_bo_get_device(bo);
if(fb->fb_id) drmModeRmFB(drm.fd, fb->fb_id);
free(fb);
}
static struct drm_fb * drm_fb_get_from_bo(struct gbm_bo *bo) {
struct drm_fb *fb = gbm_bo_get_user_data(bo);
uint32_t width, height, stride, handle;
int ret;
if(fb)
return fb;
fb = calloc(1, sizeof *fb);
fb->bo = bo;
width = gbm_bo_get_width(bo);
height = gbm_bo_get_height(bo);
stride = gbm_bo_get_stride(bo);
handle = gbm_bo_get_handle(bo).u32;
ret = drmModeAddFB(drm.fd, width, height, 24, 32, stride, handle, &fb->fb_id);
if(ret) {
printf("failed to create fb: %s\n", strerror(errno));
free(fb);
return NULL;
}
gbm_bo_set_user_data(bo, fb, drm_fb_destroy_callback);
return fb;
}
static void page_flip_handler(
int fd, unsigned int frame, unsigned int sec, unsigned int usec, void *data
) {
int *waiting_for_flip = data;
*waiting_for_flip = 0;
}
static int32_t drmInit(void) { static int32_t drmInit(void) {
drmModeRes *resources; drmModeRes *resources;
drmModeConnector *connector = NULL; drmModeConnector *connector = NULL;
@ -123,9 +166,7 @@ static int32_t drmInit(void) {
// Find an existing connected connector. // Find an existing connected connector.
for (i = 0; i < resources->count_connectors; i++) { for (i = 0; i < resources->count_connectors; i++) {
connector = drmModeGetConnector(drm.fd, resources->connectors[i]); connector = drmModeGetConnector(drm.fd, resources->connectors[i]);
if(connector->connection == DRM_MODE_CONNECTED) { if(connector->connection == DRM_MODE_CONNECTED) break;
break;
}
drmModeFreeConnector(connector); drmModeFreeConnector(connector);
connector = NULL; connector = NULL;
} }
@ -143,7 +184,7 @@ static int32_t drmInit(void) {
if(current_mode->type & DRM_MODE_TYPE_PREFERRED) drm.mode = current_mode; if(current_mode->type & DRM_MODE_TYPE_PREFERRED) drm.mode = current_mode;
int current_area = current_mode->hdisplay * current_mode->vdisplay; int current_area = current_mode->hdisplay * current_mode->vdisplay;
if(current_area >= area) { if(current_area > area) {
drm.mode = current_mode; drm.mode = current_mode;
area = current_area; area = current_area;
} }
@ -187,7 +228,7 @@ static int32_t gbmInit(void) {
gbm.surface = gbm_surface_create( gbm.surface = gbm_surface_create(
gbm.dev, gbm.dev,
drm.mode->vdisplay, drm.mode->vdisplay, drm.mode->hdisplay, drm.mode->vdisplay,
GBM_FORMAT_XRGB8888, GBM_FORMAT_XRGB8888,
GBM_BO_USE_SCANOUT | GBM_BO_USE_RENDERING GBM_BO_USE_SCANOUT | GBM_BO_USE_RENDERING
); );
@ -270,47 +311,20 @@ static int32_t glInit(void) {
return 0; return 0;
} }
void gbmUpdateResolution(game_t *game) {
// Get the surface dimensions
EGLint width;
EGLint height;
eglQuerySurface(gl.display, gl.surface, EGL_WIDTH,&width);
eglQuerySurface(gl.display, gl.surface, EGL_HEIGHT,&height);
static void drm_fb_destroy_callback(struct gbm_bo *bo, void *data) { bool needsUpdating = (
struct drm_fb *fb = data; (width != game->engine.render.width) ||
struct gbm_device *gbm = gbm_bo_get_device(bo); (height != game->engine.render.height
);
if(fb->fb_id) drmModeRmFB(drm.fd, fb->fb_id); if(!needsUpdating) return;
free(fb); renderSetResolution(&game->engine.render, (int32_t)width, (int32_t)height);
}
static struct drm_fb * drm_fb_get_from_bo(struct gbm_bo *bo) {
struct drm_fb *fb = gbm_bo_get_user_data(bo);
uint32_t width, height, stride, handle;
int ret;
if(fb)
return fb;
fb = calloc(1, sizeof *fb);
fb->bo = bo;
width = gbm_bo_get_width(bo);
height = gbm_bo_get_height(bo);
stride = gbm_bo_get_stride(bo);
handle = gbm_bo_get_handle(bo).u32;
ret = drmModeAddFB(drm.fd, width, height, 24, 32, stride, handle, &fb->fb_id);
if(ret) {
printf("failed to create fb: %s\n", strerror(errno));
free(fb);
return NULL;
}
gbm_bo_set_user_data(bo, fb, drm_fb_destroy_callback);
return fb;
}
static void page_flip_handler(
int fd, unsigned int frame, unsigned int sec, unsigned int usec, void *data
) {
int *waiting_for_flip = data;
*waiting_for_flip = 0;
} }
int32_t main(int argc, char *argv[]) { int32_t main(int argc, char *argv[]) {
@ -322,7 +336,6 @@ int32_t main(int argc, char *argv[]) {
struct gbm_bo *bo; struct gbm_bo *bo;
struct drm_fb *fb; struct drm_fb *fb;
int32_t ret; int32_t ret;
int32_t width, height;
// Init the DRM // Init the DRM
ret = drmInit(); ret = drmInit();
@ -356,9 +369,6 @@ int32_t main(int argc, char *argv[]) {
// Now uh? // Now uh?
bo = gbm_surface_lock_front_buffer(gbm.surface); bo = gbm_surface_lock_front_buffer(gbm.surface);
fb = drm_fb_get_from_bo(bo); fb = drm_fb_get_from_bo(bo);
width = gbm_bo_get_width(bo);
height = gbm_bo_get_height(bo);
stride = gbm_bo_get_stride(bo);
// ? // ?
ret = drmModeSetCrtc( ret = drmModeSetCrtc(
@ -372,8 +382,9 @@ int32_t main(int argc, char *argv[]) {
// Init Game // Init Game
game_t *game = malloc(sizeof(game_t)); game_t *game = malloc(sizeof(game_t));
printf("Game is %zu bytes.\n", sizeof(game_t)); printf("Game is %zu bytes.\n", sizeof(game_t));
printf("Resolution is %i x %i x %i\n", width, height, stride); gbmUpdateResolution(game);
renderSetResolution(&game->engine.render, (int32_t)width, (int32_t)height);
// Get the surface dimensions
if(!gameInit(game)) { if(!gameInit(game)) {
printf("Game Init Error\n"); printf("Game Init Error\n");
return 1; return 1;
@ -385,6 +396,7 @@ int32_t main(int argc, char *argv[]) {
// Draw // Draw
float fDelta = 0.016f; float fDelta = 0.016f;
gbmUpdateResolution(game);
if(!gameUpdate(game, fDelta)) break; if(!gameUpdate(game, fDelta)) break;
// Drawn // Drawn