掌機 - Z-Pocket Game Pro (ZPG Pro) - libdrm(Page-Flip)



參考資訊:
https://blog.csdn.net/hexiaolong2009/article/details/84674127

main.c

#define _GNU_SOURCE
#include <errno.h>
#include <fcntl.h>
#include <stdbool.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/mman.h>
#include <time.h>
#include <unistd.h>
#include <signal.h>
#include <xf86drm.h>
#include <xf86drmMode.h>
 
struct fb_obj {
    uint32_t width;
    uint32_t height;
    uint32_t pitch;
    uint32_t handle;
    uint32_t size;
    uint32_t *vaddr;
    uint32_t fb_id;
};
 
static int terminate=0;
struct fb_obj fb[2]={0};
 
static int modeset_create_fb(int fd, struct fb_obj *bo, uint32_t color)
{
    uint32_t i=0;
    struct drm_mode_map_dumb map={0};
    struct drm_mode_create_dumb create={0};
 
    create.width = bo->width;
    create.height = bo->height;
    create.bpp = 32;
    drmIoctl(fd, DRM_IOCTL_MODE_CREATE_DUMB, &create);
    bo->pitch = create.pitch;
    bo->size = create.size;
    bo->handle = create.handle;
    drmModeAddFB(fd, bo->width, bo->height, 24, 32, bo->pitch, bo->handle, &bo->fb_id);
    map.handle = create.handle;
    drmIoctl(fd, DRM_IOCTL_MODE_MAP_DUMB, &map);
    bo->vaddr = mmap(0, create.size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, map.offset);
    for(i=0; i<(bo->size/4); i++){
        bo->vaddr[i] = color;
    }
    return 0;
}
 
static void modeset_destroy_fb(int fd, struct fb_obj *bo)
{
    struct drm_mode_destroy_dumb destroy={0};
 
    drmModeRmFB(fd, bo->fb_id);
    munmap(bo->vaddr, bo->size);
    destroy.handle = bo->handle;
    drmIoctl(fd, DRM_IOCTL_MODE_DESTROY_DUMB, &destroy);
}
 
static void modeset_page_flip_handler(int fd, uint32_t frame, uint32_t sec, uint32_t usec, void *data)
{
    static int i=0;
    uint32_t crtc_id=*(uint32_t*)data;
 
    i^= 1;
    drmModePageFlip(fd, crtc_id, fb[i].fb_id, DRM_MODE_PAGE_FLIP_EVENT, data);
    usleep(500000);
}
 
static void sigint_handler(int arg)
{
    terminate = 1;
}
 
int main(int argc, char **argv)
{
    int fd=-1;
    uint32_t conn_id=-1;
    uint32_t crtc_id=-1;
    drmModeRes *res=NULL;
    drmEventContext ev={0};
    drmModeConnector *conn=NULL;
 
    signal(SIGINT, sigint_handler);
    ev.version = DRM_EVENT_CONTEXT_VERSION;
    ev.page_flip_handler = modeset_page_flip_handler;
    fd = open("/dev/dri/card0", O_RDWR | O_CLOEXEC);
    res = drmModeGetResources(fd);
    crtc_id = res->crtcs[0];
    conn_id = res->connectors[0];
    conn = drmModeGetConnector(fd, conn_id);
    fb[0].width = conn->modes[0].hdisplay;
    fb[0].height = conn->modes[0].vdisplay;
    fb[1].width = conn->modes[0].hdisplay;
    fb[1].height = conn->modes[0].vdisplay;
    modeset_create_fb(fd, &fb[0], 0xff0000);
    modeset_create_fb(fd, &fb[1], 0x0000ff);
    drmModeSetCrtc(fd, crtc_id, fb[0].fb_id, 0, 0, &conn_id, 1, &conn->modes[0]);
    drmModePageFlip(fd, crtc_id, fb[0].fb_id, DRM_MODE_PAGE_FLIP_EVENT, &crtc_id);
    while(!terminate){
        drmHandleEvent(fd, &ev);
    }
    modeset_destroy_fb(fd, &fb[1]);
    modeset_destroy_fb(fd, &fb[0]);
    drmModeFreeConnector(conn);
    drmModeFreeResources(res);
    close(fd);

    return 0;
}

編譯

$ sudo apt-get install libdrm-dev -y
$ gcc main.c -o test -I/usr/include/libdrm -ldrm