Edit online

Demo

28 Nov 2024
Read time: 25 minute(s)

Framebuffer 标准接口的使用

标准 framebuffer 接口操作示例:
int main()
{
    int fd = 0;
    struct fb_var_screeninfo vi;
    struct fb_fix_screeninfo fi;

    void *fb_ptr = NULL;

    fd = open("/dev/fb0", O_RDWR);
    if (fd < 0) {
        printf("failed to open fb0\n");
        return -1;
    }

    // 获取可变屏幕参数,如分辨率,像素格式
    if (ioctl(fd, FBIOGET_VSCREENINFO, &vi) < 0) {
        printf("failed to get fb0 info\n");
        close(fd);
        return -1;
    }

    // 获取屏幕固定参数,包括显存起始物理地址、显存大小和行间距
    if (ioctl(fd, FBIOGET_FSCREENINFO, &fi) < 0) {
        printf("failed to get fb0 info\n");
        close(fd);
        return -1;
    }

    // framebuffer 缓冲区映射到用户空间:
    fb_ptr = mmap(0, fi.smem_len, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
    if (fb_ptr == MAP_FAILED) {
        printf("failed to mmap framebuffer\n");
        close(fd);
        return -1;
    }

    // 用户可以直接把要显示到屏幕的图像数据写入 ptr 即可
    ........
    ........

    munmap(fb_ptr, fi.smem_len);
    close(fd);

    return 0;
}

AICFB 扩展接口的使用

代码详见 source/artinchip/test-fb/test_fb.c
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Copyright (C) 2020-2021 ArtInChip Technology Co., Ltd.
 * Authors:  Matteo <duanmt@artinchip.com>
 */

#include "base.h"
#include "artinchip_fb.h"

/* Global macro and variables */

#define AICFB_LAYER_MAX_NUM 2
#define FB_DEV "/dev/fb0"

static const char sopts[] = "nscflLaAkKedw:h:m:v:u";
static const struct option lopts[] = {
    {"get_layer_num",   no_argument, NULL, 'n'},
    {"get_screen_size", no_argument, NULL, 's'},
    {"get_layer_cap",   no_argument, NULL, 'c'},
    {"get_fb_layer",    no_argument, NULL, 'f'},
    {"get_layer",       no_argument, NULL, 'l'},
    {"set_layer",       no_argument, NULL, 'L'},
    {"get_alpha",       no_argument, NULL, 'a'},
    {"set_alpha",     required_argument, NULL, 'A'},
    {"get_ck_cfg",      no_argument, NULL, 'k'},
    {"set_ck_cfg",      no_argument, NULL, 'K'},
    {"enable",      no_argument, NULL, 'e'},
    {"disable",     no_argument, NULL, 'd'},
    {"id",        required_argument, NULL, 'i'},
    {"width",     required_argument, NULL, 'w'},
    {"height",    required_argument, NULL, 'h'},
    {"mode",      required_argument, NULL, 'm'},
    {"value",     required_argument, NULL, 'v'},
    {"usage",       no_argument, NULL, 'u'},
    {0, 0, 0, 0}
};

/* Functions */

void usage(char *program)
{
    printf("Usage: %s [options]: \n", program);
    printf("\t -n, --get_layer_num \n");
    printf("\t -s, --get_screen_size \n");
    printf("\t -c, --get_layer_cap \n");
    printf("\t -f, --get_fb_layer \n");
    printf("\t -l, --get_layer \n");
    printf("\t -L, --set_layer\tneed other options: -i x -e/d -w y -h z\n");
    printf("\t -a, --get_alpha \n");
    printf("\t -A, --set_alpha\tneed other options: -e/d -m x -v y \n");
    printf("\t -k, --get_ck_cfg \n");
    printf("\t -K, --set_ck_cfg\tneed other options: -e/d -v x \n");
    printf("\t -e, --enable \n");
    printf("\t -d, --disable \n");
    printf("\t -i, --id\t\tneed an integer argument of Layer ID\n");
    printf("\t -w, --width\t\tneed an integer argument\n");
    printf("\t -h, --height\t\tneed an integer argument\n");
    printf("\t -m, --mode\t\tneed an integer argument\n");
    printf("\t -v, --value\t\tneed an integer argument\n");
    printf("\t -u, --usage \n");
    printf("\n");
    printf("Example: %s -l\n", program);
    printf("Example: %s -L -i 1 -e -w 800 -h 480\n", program);
    printf("Example: %s -A -e -w 800 -h 480\n", program);
    printf("Example: %s -K -e -v 0x3F\n", program);
}

/* Open a device file to be needed. */
int device_open(char *_fname, int _flag)
{
    s32 fd = -1;

    fd = open(_fname, _flag);
    if (fd < 0) {
        ERR("Failed to open %s", _fname);
        exit(0);
    }
    return fd;
}

int get_layer_num(int fd)
{
    int ret = 0;
    struct aicfb_layer_num n = {0};

    ret = ioctl(fd, AICFB_GET_LAYER_NUM, &n);
    if (ret < 0) {
        ERR("ioctl() return %d\n", ret);
    }
    else {
        printf("The number of video layer: %d\n", n.vi_num);
        printf("The number of UI layer: %d\n", n.ui_num);
    }
    return ret;
}

int get_layer_cap(int fd)
{
    int i;
    int ret = 0;
    struct aicfb_layer_capability cap = {0};

    for (i = 0; i < AICFB_LAYER_MAX_NUM; i++) {
        memset(&cap, 0, sizeof(struct aicfb_layer_capability));
        cap.layer_id = i;
        ret = ioctl(fd, AICFB_GET_LAYER_CAPABILITY, &cap);
        if (ret < 0) {
            ERR("ioctl() return %d\n", ret);
            return ret;
        }

        printf("\n--------------- Layer %d ---------------\n", i);
        printf("Type: %s\n", cap.layer_type == AICFB_LAYER_TYPE_VIDEO ?
            "Video" : "UI");
        printf("Max Width: %d (%#x)\n", cap.max_width, cap.max_width);
        printf("Max height: %d (%#x)\n", cap.max_height, cap.max_height);
        printf("Flag: %#x\n", cap.cap_flags);
        printf("---------------------------------------\n");
    }
    return ret;
}

int get_one_layer_cfg(int fd, int cmd, int id)
{
    int ret = 0;
    struct aicfb_layer_data layer = {0};

    layer.layer_id = id;
    ret = ioctl(fd, cmd, &layer);
    if (ret < 0) {
        ERR("ioctl() return %d\n", ret);
        return ret;
    }

    printf("\n--------------- Layer %d ---------------\n", layer.layer_id);
    printf("Enable: %d\n", layer.enable);
    printf("Layer ID: %d\n", layer.layer_id);
    printf("Rect ID: %d\n", layer.rect_id);
    printf("Scale Size: w %d, h %d\n",
        layer.scale_size.width, layer.scale_size.height);
    printf("Position: x %d, y %d\n", layer.pos.x, layer.pos.y);
    printf("Buffer: \n");
    printf("\tPixel format: %#x\n", layer.buf.format);
    printf("\tPhysical addr: %#x %#x %#x\n",
        layer.buf.phy_addr[0], layer.buf.phy_addr[1],
        layer.buf.phy_addr[2]);
    printf("\tStride: %d %d %d\n", layer.buf.stride[0],
        layer.buf.stride[1], layer.buf.stride[2]);
    printf("\tSize: w %d, h %d\n",
        layer.buf.size.width, layer.buf.size.height);
    printf("\tCrop enable: %d\n", layer.buf.crop_en);
    printf("\tCrop: x %d, y %d, w %d, h %d\n",
        layer.buf.crop.x, layer.buf.crop.y,
        layer.buf.crop.width, layer.buf.crop.height);
    printf("\tFlag: %#x\n", layer.buf.buf_flags);
    printf("---------------------------------------\n");

    return 0;
}

int get_fb_layer_cfg(int fd)
{
    return get_one_layer_cfg(fd, AICFB_GET_FB_LAYER_CONFIG, 0);
}

int get_layer_cfg(int fd)
{
    int i;

    for (i = 0; i < AICFB_LAYER_MAX_NUM; i++)
        get_one_layer_cfg(fd, AICFB_GET_LAYER_CONFIG, i);

    return 0;
}

int set_layer_cfg(int fd, int id, int enable, int width, int height)
{
    int ret = 0;
    struct aicfb_layer_data layer = {0};

    if ((id < 0) || (enable < 0) || (width < 0) || (height < 0)) {
        ERR("Invalid argument.\n");
        return -1;
    }

    layer.layer_id = id;
    layer.enable = enable;
    layer.scale_size.width = layer.buf.size.width;
    layer.scale_size.height = layer.buf.size.height;
    ret = ioctl(fd, AICFB_UPDATE_LAYER_CONFIG, &layer);
    if (ret < 0)
        ERR("ioctl() return %d\n", ret);

    return ret;
}

int get_layer_alpha(int fd)
{
    int i;
    int ret = 0;
    struct aicfb_alpha_config alpha = {0};

    for (i = 1; i < AICFB_LAYER_MAX_NUM; i++) {
        alpha.layer_id = i;
        ret = ioctl(fd, AICFB_GET_ALPHA_CONFIG, &alpha);
        if (ret < 0) {
            ERR("ioctl() return %d\n", ret);
            return ret;
        }

        printf("\n--------------- Layer %d ---------------\n", i);
        printf("Alpha enable: %d\n", alpha.enable);
        printf("Alpla mode: %d (0, pixel; 1, global; 2, mix)\n",
            alpha.mode);
        printf("Alpha value: %d (%#x)\n", alpha.value, alpha.value);
        printf("---------------------------------------\n");
    }

    return 0;
}

int set_layer_alpha(int fd, int enable, int mode, int val)
{
    int ret = 0;
    struct aicfb_alpha_config alpha = {0};

    if ((enable < 0) || (mode < 0) || (val < 0)) {
        ERR("Invalid argument.\n");
        return -1;
    }

    alpha.layer_id = 1;
    alpha.enable = enable;
    alpha.mode = mode;
    alpha.value = val;
    ret = ioctl(fd, AICFB_UPDATE_ALPHA_CONFIG, &alpha);
    if (ret < 0)
        ERR("ioctl() return %d\n", ret);

    return ret;
}

int get_ck_cfg(int fd)
{
    int i;
    int ret = 0;
    struct aicfb_ck_config ck = {0};

    for (i = 1; i < AICFB_LAYER_MAX_NUM; i++) {
        ck.layer_id = i;
        ret = ioctl(fd, AICFB_GET_CK_CONFIG, &ck);
        if (ret < 0) {
            ERR("ioctl() return %d\n", ret);
            return ret;
        }

        printf("\n--------------- Layer %d ---------------\n", i);
        printf("Color key enable: %d\n", ck.enable);
        printf("Color key value: R %#x, G %#x, B %#x\n",
            (ck.value >> 16) & 0xFF, (ck.value >> 8) & 0xFF,
            ck.value & 0xFF);
        printf("---------------------------------------\n");
    }
    return 0;
}

int set_ck_cfg(int fd, int enable, int val)
{
    int ret = 0;
    struct aicfb_ck_config ck = {0};

    if ((enable < 0) || (val < 0)) {
        ERR("Invalid argument.\n");
        return -1;
    }

    ck.layer_id = 1;
    ck.enable = enable;
    ck.value = val;
    ret = ioctl(fd, AICFB_UPDATE_CK_CONFIG, &ck);
    if (ret < 0)
        ERR("ioctl() return %d\n", ret);

    return ret;
}

int get_screen_size(int fd)
{
    int ret = 0;
    struct aic_size s = {0};

    ret = ioctl(fd, AICFB_GET_SCREEN_SIZE, &s);
    if (ret < 0) {
        ERR("ioctl() return %d\n", ret);
        return ret;
    }

    printf("Screen width: %d (%#x)\n", s.width, s.width);
    printf("Screen height: %d (%#x)\n", s.height, s.height);
    return 0;
}

int main(int argc, char **argv)
{
    int dev_fd = -1;
    int ret = 0;
    int c = 0;
    int layer_id = 0;
    int enable = 0;
    int mode = 0;
    int width = 0;
    int height = 0;
    int value = 0;

    dev_fd = device_open(FB_DEV, O_RDWR);
    if (dev_fd < 0) {
        ERR("Failed to open %s, return %d\n", FB_DEV, dev_fd);
        return -1;
    }

    while ((c = getopt_long(argc, argv, sopts, lopts, NULL)) != -1) {
        switch (c) {
        case 'n':
            ret = get_layer_num(dev_fd);
            goto end;
        case 's':
            ret = get_screen_size(dev_fd);
            goto end;
        case 'c':
            ret = get_layer_cap(dev_fd);
            goto end;
        case 'f':
            ret = get_fb_layer_cfg(dev_fd);
            goto end;
        case 'l':
            ret = get_layer_cfg(dev_fd);
            goto end;
        case 'a':
            ret = get_layer_alpha(dev_fd);
            goto end;
        case 'k':
            ret = get_ck_cfg(dev_fd);
            goto end;
        case 'e':
            enable = 1;
            continue;
        case 'd':
            enable = 0;
            continue;
        case 'i':
            layer_id = str2int(optarg);
            continue;
        case 'w':
            width = str2int(optarg);
            continue;
        case 'h':
            height = str2int(optarg);
            continue;
        case 'm':
            mode = str2int(optarg);
            continue;
        case 'v':
            value = str2int(optarg);
            continue;
        case 'u':
            usage(argv[0]);
            goto end;
        default:
            continue;
        }
    }

    optind = 0;
    while ((c = getopt_long(argc, argv, sopts, lopts, NULL)) != -1) {
        switch (c) {
        case 'L':
            ret = set_layer_cfg(dev_fd, layer_id, enable, width, height);
            goto end;
        case 'A':
            ret = set_layer_alpha(dev_fd, enable, mode, value);
            goto end;
        case 'K':
            ret = set_ck_cfg(dev_fd, enable, value);
            goto end;
        default:
            continue;
        }
    }

end:
    if (dev_fd > 0)
        close(dev_fd);

    return ret;
}

DMA-BUF 的使用

代码详见 source/artinchip/test-dma-buf/
// SPDX-License-Identifier: GPL-2.0-only
/*
 * Copyright (C) 2020-2021 ArtInChip Technology Co., Ltd.
 * Authors:  Matteo <duanmt@artinchip.com>
 */

#include <artinchip/sample_base.h>
#include <linux/dma-buf.h>
#include <linux/dma-heap.h>
#include <video/artinchip_fb.h>

/* Global macro and variables */

#define AICFB_VID_BUF_NUM   2
#define AIC_CMA_BUF_MAX     (8 * 1024 * 1024)
#define DMA_HEAP_DEV        "/dev/dma_heap/reserved"
#define FB_DEV          "/dev/fb0"

static const char sopts[] = "m:w:h:f:i:u";
static const struct option lopts[] = {
    {"width",     required_argument, NULL, 'w'},
    {"height",    required_argument, NULL, 'h'},
    {"format",    required_argument, NULL, 'f'},
    {"input",     required_argument, NULL, 'i'},
    {"usage",       no_argument, NULL, 'u'},
    {0, 0, 0, 0}
};

struct video_data_format {
    enum aic_pixel_format format;
    char f_str[12];
    int y_shift;
    int u_shift;
    int v_shift;
};

struct video_data_format g_vformat[] = {
    {AIC_FMT_YUV420P, "yuv420p", 0, 2, 2},
    {AIC_FMT_YUV422P, "yuv422p", 0, 1, 1},
    {AIC_FMT_YUV444P, "yuv444p", 0, 0, 0},

    {AIC_FMT_MAX, "", 0, 0, 0}
};

struct video_plane {
    int fd;
    char *buf;
    int len;
};

struct video_buf {
    struct video_plane y;
    struct video_plane u;
    struct video_plane v;
};

struct aicfb_video_layer {
    int w;
    int h;
    struct video_data_format *f;
    struct video_buf vbuf[AICFB_VID_BUF_NUM];
};

static int g_fb_fd = -1;
static struct aicfb_video_layer g_vlayer = {0};

/* Functions */

void usage(char *program)
{
    printf("Usage: %s [options]: \n", program);
    printf("\t -w, --width\t\tneed an integer argument\n");
    printf("\t -h, --height\t\tneed an integer argument\n");
    printf("\t -f, --format\t\tvideo format, yuv420p etc\n");
    printf("\t -i, --input\t\tneed a file name \n");
    printf("\t -u, --usage \n");
    printf("\n");
    printf("Example: %s -w 480 -h 320 -f yuv420p -i my.yuv\n", program);
}

/* Open a device file to be needed. */
int device_open(char *_fname, int _flag)
{
    s32 fd = -1;

    fd = open(_fname, _flag);
    if (fd < 0) {
        ERR("Failed to open %s", _fname);
        exit(0);
    }
    return fd;
}

int vidbuf_request_one(struct video_plane *plane, int len, int fd)
{
    int ret;
    char *buf = NULL;
    struct dma_heap_allocation_data data = {0};

    if ((len < 0) || (len > AIC_CMA_BUF_MAX)) {
        ERR("Invalid len %d\n", len);
        return -1;
    }

    data.fd = 0;
    data.len = len;
    data.fd_flags = O_RDWR;
    data.heap_flags = 0;
    ret = ioctl(fd, DMA_HEAP_IOCTL_ALLOC, &data);
    if (ret < 0) {
        ERR("ioctl() failed! errno: %d[%s]\n", errno, strerror(errno));
        return -1;
    }

    DBG("Get dma_heap fd: %d\n", data.fd);
    plane->fd = data.fd;

    buf = mmap(NULL, len, PROT_READ|PROT_WRITE, MAP_SHARED, data.fd, 0);
    if (buf == MAP_FAILED) {
        ERR("mmap() failed! errno: %d[%s]\n", errno, strerror(errno));
        return -1;
    }
    DBG("Get virtual address: %p\n", buf);
    plane->buf = buf;
    plane->len = len;
    return 0;
}

void aic_fb_open(void)
{
    if (g_fb_fd > 0)
        return;

    g_fb_fd = device_open(FB_DEV, O_RDWR);
    if (g_fb_fd < 0)
        ERR("Failed to open %s. errno: %d[%s]\n",
            FB_DEV, errno, strerror(errno));
}

int set_ui_layer_alpha(int val)
{
    int ret = 0;
    struct aicfb_alpha_config alpha = {0};

    alpha.layer_id = 1;
    alpha.enable = 1;
    alpha.mode = 1;
    alpha.value = val;
    ret = ioctl(g_fb_fd, AICFB_UPDATE_ALPHA_CONFIG, &alpha);
    if (ret < 0)
        ERR("ioctl() failed! errno: %d[%s]\n", errno, strerror(errno));

    return ret;
}

int vidbuf_request(struct aicfb_video_layer *vlayer)
{
    int i, j;
    int heap_fd = -1;
    int y_frame = vlayer->w * vlayer->h;

    heap_fd = device_open(DMA_HEAP_DEV, O_RDWR);
    if (heap_fd < 0) {
        ERR("Failed to open %s, errno: %d[%s]\n",
            DMA_HEAP_DEV, errno, strerror(errno));
        return -1;
    }

    /* Prepare two group buffer for video player,
       and each group has three planes: y, u, v. */
    for (i = 0; i < AICFB_VID_BUF_NUM; i++) {
        struct video_plane *p = (struct video_plane *)&vlayer->vbuf[i];
        int *shift = &vlayer->f->y_shift;
        for (j = 0; j < AICFB_PLANE_NUM; j++, p++)
            vidbuf_request_one(p, y_frame >> shift[j], heap_fd);
    }

    close(heap_fd);
    return 0;
}

void vidbuf_release(struct aicfb_video_layer *vlayer)
{
    int i, j;

    for (i = 0; i < AICFB_VID_BUF_NUM; i++) {
        struct video_plane *p = (struct video_plane *)&vlayer->vbuf[i];
        for (j = 0; j < AICFB_PLANE_NUM; j++, p++) {
            if (munmap(p->buf, p->len) < 0)
                ERR("munmap() failed! errno: %d[%s]\n",
                    errno, strerror(errno));
        }
    }
}

void vidbuf_dmabuf_begin(struct aicfb_video_layer *vlayer)
{
    int i, j;
    struct aicfb_dmabuf_fd fds = {0};

    for (i = 0; i < AICFB_VID_BUF_NUM; i++) {
        struct video_plane *plane = (struct video_plane *)&vlayer->vbuf[i];
        for (j = 0; j < AICFB_PLANE_NUM; j++, plane++) {
            fds.fd = plane->fd;
            if (ioctl(g_fb_fd, AICFB_GET_DMABUF, &fds) < 0)
                ERR("ioctl() failed! err %d[%s]\n",
                    errno, strerror(errno));
        }
    }
}

void vidbuf_dmabuf_end(struct aicfb_video_layer *vlayer)
{
    int i, j;
    struct aicfb_dmabuf_fd fds = {0};

    for (i = 0; i < AICFB_VID_BUF_NUM; i++) {
        struct video_plane *plane = (struct video_plane *)&vlayer->vbuf[i];
        for (j = 0; j < AICFB_PLANE_NUM; j++, plane++) {
            fds.fd = plane->fd;
            if (ioctl(g_fb_fd, AICFB_PUT_DMABUF, &fds) < 0)
                ERR("ioctl() failed! err %d[%s]\n",
                    errno, strerror(errno));
        }
    }
}

void video_layer_set(struct aicfb_video_layer *vlayer, int index)
{
    struct aicfb_layer_data layer = {0};
    struct video_buf *vbuf = &vlayer->vbuf[index];

    layer.layer_id = 0;
    layer.enable = 1;
    layer.scale_size.width = vlayer->w * 4;
    layer.scale_size.height = vlayer->h * 4;
    layer.pos.x = 10;
    layer.pos.y = 10;
    layer.buf.size.width = vlayer->w;
    layer.buf.size.height = vlayer->h;
    layer.buf.format = vlayer->f->format;
    layer.buf.dmabuf_fd[0] = vbuf->y.fd;
    layer.buf.dmabuf_fd[1] = vbuf->u.fd;
    layer.buf.dmabuf_fd[2] = vbuf->v.fd;
    layer.buf.stride[0] = vlayer->w;
    layer.buf.stride[1] = vlayer->w >> 1;
    layer.buf.stride[2] = vlayer->w >> 1;

    if (ioctl(g_fb_fd, AICFB_UPDATE_LAYER_CONFIG, &layer) < 0)
        ERR("ioctl() failed! err %d[%s]\n", errno, strerror(errno));
}

void vidbuf_cpu_begin(struct video_buf *vbuf)
{
    int i;
    struct dma_buf_sync flag;
    struct video_plane *p = (struct video_plane *)vbuf;

    for (i = 0; i < AICFB_PLANE_NUM; i++, p++) {
        flag.flags = DMA_BUF_SYNC_WRITE | DMA_BUF_SYNC_START;
        if (ioctl(p->fd, DMA_BUF_IOCTL_SYNC, &flag) < 0)
            ERR("ioctl() failed! err %d[%s]\n",
                errno, strerror(errno));
    }
}

void vidbuf_cpu_end(struct video_buf *vbuf)
{
    int i;
    struct dma_buf_sync flag;
    struct video_plane *p = (struct video_plane *)vbuf;

    for (i = 0; i < AICFB_PLANE_NUM; i++, p++) {
        flag.flags = DMA_BUF_SYNC_WRITE | DMA_BUF_SYNC_END;
        if (ioctl(p->fd, DMA_BUF_IOCTL_SYNC, &flag) < 0)
            ERR("ioctl() failed! err %d[%s]\n",
                errno, strerror(errno));
    }
}

int vidbuf_read(struct aicfb_video_layer *vlayer, int index, int fd)
{
    int i, ret;
    static int frame_cnt = 0;
    struct video_plane *p = (struct video_plane *)&vlayer->vbuf[index];

    if (frame_cnt == 0)
        lseek(fd, 0, SEEK_SET);

    for (i = 0; i < AICFB_PLANE_NUM; i++, p++) {
        DBG("Frame %d - %d, offset %ld, len %d\n", frame_cnt, i,
             lseek(fd, 0, SEEK_CUR), p->len);
        ret = read(fd, p->buf, p->len);
        if (ret != p->len) {
            ERR("read(%d) return %d. errno: %d[%s]\n", p->len,
                ret, errno, strerror(errno));
            return -1;
        }
    }
    frame_cnt++;
    return ret;
}

int format_parse(char *str)
{
    int i;

    for (i = 0; g_vformat[i].format != AIC_FMT_MAX; i++) {
        if (strncasecmp(g_vformat[i].f_str, str, strlen(str)) == 0)
            return i;
    }

    ERR("Invalid format: %s\n", str);
    return 0;
}

int main(int argc, char **argv)
{
    int vid_fd = -1;
    int ret = 0;
    int c = 0;
    int fsize = 0;
    int index = 0;

    DBG("Compile time: %s\n", __TIME__);
    while ((c = getopt_long(argc, argv, sopts, lopts, NULL)) != -1) {
        switch (c) {
        case 'w':
            g_vlayer.w = str2int(optarg);
            continue;
        case 'h':
            g_vlayer.h = str2int(optarg);
            continue;
        case 'f':
            g_vlayer.f = &g_vformat[format_parse(optarg)];
            break;
        case 'i':
            vid_fd = device_open(optarg, O_RDONLY);
            if (vid_fd < 0) {
                ERR("Failed to open %s. errno: %d[%s]\n",
                    optarg, errno, strerror(errno));
                return -1;
            }
            fsize = lseek(vid_fd, 0, SEEK_END);
            DBG("open(%s) %d, size %d\n", optarg, vid_fd, fsize);
            break;
        case 'u':
            usage(argv[0]);
            return 0;
        default:
            break;
        }
    }

    aic_fb_open();
    set_ui_layer_alpha(128);
    vidbuf_request(&g_vlayer);
    vidbuf_dmabuf_begin(&g_vlayer);

    do {
        struct video_buf *vbuf = &g_vlayer.vbuf[index];

        vidbuf_cpu_begin(vbuf);
        ret = vidbuf_read(&g_vlayer, index, vid_fd);
        vidbuf_cpu_end(vbuf);
        if (ret < 0)
            break;

        video_layer_set(&g_vlayer, index);
        index = !index;
        usleep(40000);
        if (lseek(vid_fd, 0, SEEK_CUR) == fsize)
            break;
    } while (1);

    vidbuf_dmabuf_end(&g_vlayer);
    vidbuf_release(&g_vlayer);

    if (vid_fd > 0)
        close(vid_fd);
    if (g_fb_fd > 0)
        close (g_fb_fd);
    return ret;
}