Framebuffer应用编程

2023-05-16

一、Framebuffer介绍

在Linux系统中通过Framebuffer(帧缓冲)驱动程序来控制LCD。
Framebuffer就是一块内存,里面保存着一帧图像。Framebuffer中保存着一帧图像的每一个像素颜色值。LCD控制器周而复始地从Framebuffer中逐一取出每个像素的颜色值发送到LCD,这样LCD便能显示出图像。

假设LCD的分辨率是1024x768,每一个像素的颜色用32位来表示,那么Framebuffer的大小就是:1024x768x32/8=3145728字节。

二、确定LCD某坐标像素对应的Frambuffer地址

若LCD中存在某一点的坐标是(x,y),LCD最开始坐标是(0,0)。
(x,y)与(0,0)在Y方向距离y行,在X方向距离x个像素点。那么可以得出该点的偏移地址为(xres*bpp/8)*y + x *bpp/8(xres表示在纵方向的像素点数即分辨率)
xres * bpp/8即表示每一行所占用多少字节的内存

由偏移地址即可得出某点坐标处像素的起始地址:
(x,y)像素起始地址=fb_base+(xres*bpp/8)y + xbpp/8

bpp: bits per pixel 每个像素用多少位来表示它的颜色

fb_base是程序执行mmap后得到的Framebuffer地址

三、像素颜色的表示

颜色是由三原色构成,以下是三种表示方式
在这里插入图片描述
对于32BPP,一般只设置其中的低24位,高8位表示透明度,一般的LCD都不支持。
对于24BPP,硬件上为了方便处理,在Framebuffer中也是用32位来表示,效果跟32BPP是一样的。
对于16BPP,常用的是RGB565;很少的场合会用到RGB555,这可以通过ioctl读取驱动程序中的RGB位偏移来确定使用哪一种格式。

若想从以RGB888表示的颜色值得到以RGB565表示的值,只需要保留RGB888中相应高位即可。
比如RGB888的低八位表示蓝色,需要保留这八位的高5位即可得出RGB565中的蓝色数值。

四、API函数

1.ioctl函数

int ioctl(int fd, unsigned long request, ...);
// request表示与驱动程序交互的命令
//用不同的命令控制驱动程序输出我们需要的数据;

// … 表示可变参数arg,根据request命令,设备驱动程序返回输出的数据。

不同的驱动程序内部会实现不同的ioctl,APP可以使用各种ioctl跟驱动程序交互:可以传数据给驱动程序,也可以从驱动程序中读出数据。

2.mmap函数

void *mmap(void *addr, size_t length, int prot, int flags,int fd, off_t offset);

// 关闭内存映射
munmap(void *start, size_t lenght)
  • addr表示指定映射的內存起始地址,通常设为 NULL表示让系统自动选定地址,并在成功映射后返回该地址;
  • length表示将文件中多大的内容映射到内存中;
  • prot 表示映射区域的保护方式,可以为以下4种方式的组合
    a. PROT_EXEC 映射区域可被执行
    b. PROT_READ 映射区域可被读出
    c. PROT_WRITE 映射区域可被写入
    d. PROT_NONE 映射区域不能存取
  • Flags 表示影响映射区域的不同特性,常用的有以下两种
    a. MAP_SHARED 表示对映射区域写入的数据会复制回文件内,原来的文件会改变。
    b. MAP_PRIVATE 表示对映射区域的操作会产生一个映射文件的复制,对此区域的任何修改都不会写回原来的文件内容中。
  • 参数offset:文件映射的偏移量,通常设置为0,代表从文件最前方开始对应,offset必须是分页大小的整数倍。
  • 返回值:若成功映射,将返回指向映射的区域的指针,失败将返回-1。

五、代码实例

需要用ioctl函数获取LCD参数,在ioctl函数中用FBIOGET_VSCREENINFO获得屏幕的可变信息。

LCD驱动程序给APP提供2类参数:可变的参数fb_var_screeninfo、固定的参数fb_fix_screeninfo。编写应用程序时主要关心可变参数,它的结构体定义如下(#include <linux/fb.h>):

1 struct fb_var_screeninfo {
 2     __u32 xres;             // LCD的水平像素大小
 3     __u32 yres;             // LCD的垂直像素大小
 4     __u32 xres_virtual;     // LCD的虚拟水平像素大小
 5     __u32 yres_virtual;     // LCD的虚拟垂直像素大小
 6     __u32 xoffset;          // 水平像素偏移量
 7     __u32 yoffset;          // 垂直像素偏移量
 8 
 9     __u32 bits_per_pixel;            // 像素深度bpp
10     __u32 grayscale;        /* != 0 Graylevels instead of colors */
11 
12     struct fb_bitfield red;        /* bitfield in fb mem if true color, */
13     struct fb_bitfield green;    /* else only length is significant */
14     struct fb_bitfield blue;
15     struct fb_bitfield transp;    /* transparency            */    
16 
17     __u32 nonstd;            /* != 0 Non standard pixel format */
18 
19     __u32 activate;            /* see FB_ACTIVATE_*        */
20 
21     __u32 height;           // LCD的物理高度 mm
22     __u32 width;            // LCD的物理宽度 mm
23 
24     __u32 accel_flags;        /* (OBSOLETE) see fb_info.flags */
25 
27     __u32 pixclock;            // 像素时钟
28 
29     /* 下面是六个时序参数 */
30     __u32 left_margin;        /* time from sync to picture    */
31     __u32 right_margin;        /* time from picture to sync    */  
32     __u32 upper_margin;        /* time from sync to picture    */ 
33     __u32 lower_margin;
34     __u32 hsync_len;        /* length of horizontal sync    */
35     __u32 vsync_len;        /* length of vertical sync    */
36     
37     __u32 sync;            /* see FB_SYNC_*        */
38     __u32 vmode;            /* see FB_VMODE_*        */
39     __u32 rotate;            /* angle we rotate counter clockwise */
40     __u32 reserved[5];        /* Reserved for future compatibility */
41 };

具体代码如下:

#include <stdio.h>
#include <sys/ioctl.h>
#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <linux/fb.h>

struct fb_var_screeninfo var;
unsigned char *fb_base;
unsigned int line_width;
unsigned int pixel_width;
unsigned int screen_size;

// 描点函数
// 传入的color的格式是RGB888格式
void lcd_put_pixel(int x, int y, unsigned int color)
{
    unsigned char *pen_8 = fb_base + y*line_width+x*pixel_width;
    unsigned short *pen_16;
    unsigned int *pen_32;

    pen_16 = (unsigned short*)pen_8;
    pen_32 = (unsigned int*)pen_8;

    unsigned int red,green,blue;

    switch (var.bits_per_pixel)
   {
  
        case 16:
        {	
        	// 从color变量中把R、G、B抽出来
            red = (color>>16) & 0xff;
            green = (color>>8) & 0xff;
            blue = (color>>0) & 0xff;
            
            // 根据RGB565的格式,只保留red中的高5位、green中的高6位、blue中的高5位,组合成一个新的16位颜色值
            color = ((red>>3)<<11)|((green>>2)<<5)|(blue>>3);
            *pen_16 = color;
            break;
        }
        case 32:
        {
            *pen_32 = color;
            break;
        }
        default:
        {
            printf("can not support bits_per_pixel:%d\n",var.bits_per_pixel);
            break;
        }
   }
    
}

int main(int argc, char * * argv)
{   
    // 打开设备文件
    fd_fb = open("/dev/fb0",O_RDWR);

    if(fd_fb==-1){
        printf("open fb0 failed\n");
    }

    // 获取屏幕的可变信息
    if(ioctl(fd_fb,FBIOGET_VSCREENINFO,&var)==-1){
        printf("can not get var\n");
    }

    line_width = var.xres*var.bits_per_pixel/8;
    pixel_width = var.bits_per_pixel/8;
    screen_size = var.xres*var.yres*var.bits_per_pixel/8;

    // 映射framebuffer
    fb_base = (unsigned char*)mmap(NULL,screen_size,PROT_READ|PROT_WRITE,MAP_SHARED,fd_fb,0);
    if(fb_base==(unsigned char*)-1){
        printf("fb_base error\n");
    }

    // 清屏 设置为白色
    memset(fb_base,0xff,screen_size);
    int i;
    for(i=0;i<100;i++){
        lcd_put_pixel(var.xres/2+i, var.yres/2, 0xFF0000);
    }

    munmap(fb_base,screen_size);
    close(fd_fb);
    
    return 0;
}

运行平台:100ask-imx6ull开发板
交叉编译工具链:arm-linux-gnueabihf-gcc

在虚拟机已经开启NFS服务前提下,将编译好的文件复制到NFS目录下:

cp pixel ~/nfs_rootfs/

在ARM开发板上,若VMware使用桥接网络,使用下面命令挂载NFS

mount -t nfs -o nolock,vers=3  "ubuntu IP":/home/heavysea/nfs_rootfs  /mnt

在这里插入图片描述

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Framebuffer应用编程 的相关文章

  • ubuntu蓝牙相关问题(ubuntu 18.04 + ax210)

    装驱动 xff08 ax210的驱动好像要内核5 10以上能用 xff0c 可以参考我之前的博客升级 xff09 span class token function wget span span class token parameter
  • 嵌入式软件面试经典问题

    一 进程与线程 xff08 不同的系统资源管理方式 xff09 1 区别 进程 xff1a 资源分配的基本单位 xff0c 由一个或者多个线程组成 线程 xff1a 调度器进行调度的基本单位 xff0c 一个任务 每个进程都有自己独立的内存
  • ESP8266常用API函数总结(vscode+platformIO)

    1 COMMON 1 includePath ctrl 43 shift 43 p进入命令面板输入C C 43 43 Edit Configurations 在包含文件中选择include文件目录 2 输出监视器 在platformio i
  • C++基础总结

    1 引用和指针区别 引用变量是一个别名 xff0c 相当于定义了一个const类型变量 xff0c 但不分配空间 引用定义时必须初始化 xff0c 指针无要求 初始化时引用一个实体后不能再引用其他的 xff0c 指针可以在任何时候指向任一同
  • 局域网共享文件配置

    1 适用范围 可用于同一网络内的设备进行数据共享 xff08 同一WIFI或网线接同一路由器 xff09 2 共享文件配置步骤 xff1a 1 选择你想共享的文件 xff0c 鼠标右键单击属性 xff08 例如我这文件名为Shared xf
  • C语言细节

    C 1 char数组声明时初始化 2 uin8 t属于无符号字符型 typedef signed char int8 t typedef unsigned char uint8 t typedef short int16 t typedef
  • ESP32 stable_v5.0 API简单汇总

    SDK API https docs espressif com projects esp idf 1 GPIO amp EXTI 头文件位置 xff1a components driver include driver gpio h 二选
  • keil5[使用虚拟端口进行软件仿真串口调试]与[使用开发板串口调试]

    前言 xff1a 做串口调试试验 xff0c 写下此文章 介绍两种方式 xff0c 一种软件仿真 xff0c 一种硬件 1 虚拟端口进行软件仿真串口调试 xff1a 无需硬件 开发板 需要虚拟串口驱动 xff08 一个软件 xff09 xf
  • STM32 FreeRTOS学习——任务创建

    任务创建 任务概念 什么是任务 任务函数没有返回值 xff0c 并且参数是指针类型 很少使用 基本可以不管 void entry task void pvParameters 任务主体 xff0c 无限循环且不能返回值 while 1 任务
  • kinect v2 相机标定

    目录 ubuntu删除命令 相机标定 一些快捷键 准备工作 详细步骤 ubuntu删除命令 ubuntu中删除命令一般使用rm xff0c 但是rm误删之后 xff0c 想恢复比较麻烦 xff0c 所以在这里介绍另外一种删除方法 xff0c
  • 研究生如何读论文

    作为一个准研究生 xff0c 马上就要开始加入到读论文的大军中 xff0c 如何读论文是我们共同关心的问题 xff0c 笔者在这方面做了一些小调查 xff0c 发现台湾清华彭明辉教授的研究生手册非常有启发意义 xff0c 现摘录一下对自己很
  • 嵌入式面试资料整理

    第一章进程线程的基本概念 1 什么是进程 xff0c 线程 xff0c 有什么区别 2 多进程 多线程的优缺点 3 什么时候用进程 xff0c 什么时候用线程 4 多进程 多线程同步 xff08 通讯 xff09 的方法 5 进程线程的状态
  • MSP-EXP430F5529LP_GPIO

    为了能让学习更有动力并且坚持下去 xff0c 突然想到可以用博客的形式来记录自己的学习历程 xff0c 一方面是基于上述原因并且可以留下笔记来方便日后查看 xff0c 另一方面感觉写出来一些东西可以让学习更加具有逻辑性 xff0c 今天就让
  • 【STM32】STM32内存映射以及启动过程(超详细过程)

    一 内存映射 1 内存映射图 下图是 STM32F103xCDE 型号的内存映射图 2 内存划分 由于 STM32 是 32 位 xff0c 且其地址总线也为 32 根 xff0c 所以其理论能够寻找的地址大小为 4GB 从上图可以看出 x
  • Linux学习笔记(c):udp客户端向服务器端传输照片

    客户端 span class token macro property span class token directive keyword include span span class token string lt stdio h g
  • 立创EDA绘板记录

    今天是第一次接触EDA xff0c 下载了立创的软件画板子 简单上b站看了立创十几分钟的入门教学视频 xff0c 刚好自己手边有51单片机的开发板和一个最小系统 xff0c 就决定自己动手画一个最小系统 最小系统主要由复位电路 xff0c
  • 02-线性结构2 一元多项式的乘法与加法运算 (20 分)

    02 线性结构2 一元多项式的乘法与加法运算 20 分 设计函数分别求两个一元多项式的乘积与和 输入格式 输入分2行 xff0c 每行分别先给出多项式非零项的个数 xff0c 再以指数递降方式输入一个多项式非零项系数和指数 xff08 绝对
  • 03-树2 List Leaves (25 分)

    03 树2 List Leaves 25 分 Given a tree you are supposed to list all the leaves in the order of top down and left to right I
  • Jupyter Notebook上使用tensorflow的血泪教训

    本文所涉及的环境为CUDA Version 10 1 105 xff0c tensorflow gpu2 3 0 xff0c python3 6 xff0c 本文不涉及任何配置环境的细节 xff0c 仅分享在Jupyter Notebook
  • java.sql.SQLNonTransientConnectionException: Could not create connection to database server

    使用JdbcTemplate连接数据库时发生错误 xff1a java sql SQLNonTransientConnectionException Could not create connection to x1f4a5 报错有这么长

随机推荐

  • Springboot集成Neo4j与Shiro冲突

    springboot集成neo4j时 xff0c 由于neo4j与shiro均包含名为 34 sessionFactory 34 的bean xff0c 项目启动时在生成neo4j的session对象时 createSharedSessio
  • 2.4 开发自己的Spring Boot项目

    Spring MVC的视图解析器的作用主要就是定位视图的 xff0c 也就是当控制器只是返回一个逻辑名称的时候 xff0c 是没有办法直接找到对应视图的 xff0c 这就需要视图解析器来解析了 在Maven的pom xml中加入JSP和JS
  • 3.1 IoC容器简介

    所有的IoC容器都需要实现接口BeanFactory xff0c 这是一个顶级的容器接口 BeanFactory接口源码 xff1a span class token comment span span class token commen
  • 3.2.1 通过扫描装配你的Bean

    对于扫描装配而言使用的注解是 64 Component和 64 ComponentScan 64 Component是标明哪个类被扫描进入IoC容器 xff0c 而 64 ComponentScan则是标明采用何种策略去扫描装配Bean 修
  • 3.5 使用属性文件

    可以采用默认为我们配置的application properties xff0c 也可以使用自定义的配置文件 引入属性文件依赖 span class token tag span class token tag span class tok
  • 3.6 条件装配Bean

    Bean初始化前 xff0c 对某些属性进行校验 xff0c 满足校验才去装配数据源 为了处理这样的场景 xff0c 需要用到 64 Conditional注解 xff0c 同时需要配合另外一个接口Condition xff08 org s
  • 3.7 Bean的作用域

    isSingleton方法如果返回true xff0c 则Bean在IoC容器中以单例存在 xff0c 这也是Spring IoC容器的默认值 xff1b 如果isPrototype方法返回true xff0c 则当我们每次获取Bean的时
  • 3.8 使用@Profile

    在企业开发的过程中 xff0c 项目往往要面临开发环境 测试环境 准生产环境和生产环境的切换 xff0c 每一套的上下文是不一样的 xff0c 它们有各自的数据库资源 Profile机制 xff1a 实现各个环境之间的切换 假设存在dev
  • 3.9 引入XML配置Bean

    注解 64 ImportResource xff1a 可以引入对应的XML文件 xff0c 用以加载Bean 有时候有些框架 xff08 如Dubbo xff0c Cat xff09 是基于Spring的XML方式进行开发的 xff0c 这
  • 第2章概述

    示例代码 xff1a span class token doctype span class token punctuation lt span span class token doctype tag DOCTYPE span span
  • 【无标题】

    简单接口HelloService span class token keyword public span span class token keyword interface span span class token class nam
  • 穷举法解华为bl锁

    穷举法解华为bl锁 python3代码测试截图 灵感来自于 xff1a https blog csdn net qq 40169767 article details 90481748 但是我不懂shell脚本 xff0c 那个脚本又运行不
  • 4.1.2 ProxyBean的实现

    代理 xff0c 可控制或增加对目标对象的访问 ProxyBean代码 xff1a span class token keyword package span span class token namespace com span clas
  • IDEA中打开.properties文件乱码

    问题 xff1a IDEA中打开 properties文件乱码 解决方法 xff1a 一 代开settings设置 二 搜索File Encodings 三 在Properties Files中选择编码格式为UTF 8 xff0c 注意一定
  • 怎么避免下载流氓软件,误下载后怎么处理

    经历复盘 xff1a 2022 1 20时 xff0c 为了下载SPSS时不小心下载了流氓软件 回过头来复盘一下 xff0c 一个是因为当时不小心点击了电信高速下载 xff0c 下载了一个P2P软件 点击安装的时候就感觉不太对劲 xff0c
  • eclipse查看不了源码怎么办

    问题 使用eclipse的过程中 xff0c 发现ctrl 43 鼠标左键 查看不了源码 解决方案 可以在弹出的提示框中添加源码的压缩包 xff0c 压缩包在你安装的jdk的根目录下 xff0c 如我的是C Application java
  • 关于if循环语句的四种写法

    if语句在java中起到判断语句的作用写法有四种 xff0c 如下 xff1a if if else if else if if else if else 例如 xff1a 一 xff1a if语句第一种写法 if 好处 xff1a 简单便
  • ROS学习——2编写简单的发布者和订阅者

    ros官网教程 xff1a 编写简单的发布者和订阅者 目录 c 43 43 版python版注意 c 43 43 版 在新建好的catkin ws工作空间下建立 cd catkin ws src 在该目录下创建自己的功能包 xff1a ca
  • ubuntu文件系统结构与磁盘管理

    一 根目录 xff1a Linux下 就是根目录 所有的目录都是由根目录衍生出来的 二 ubuntu 文件系统结构 xff1a 三 绝对路径和相对路径 xff1a 绝对路径 xff1a 从根目录 算起的路径 相对路径 xff1a 相对于目前
  • Framebuffer应用编程

    一 Framebuffer介绍 在Linux系统中通过Framebuffer xff08 帧缓冲 xff09 驱动程序来控制LCD Framebuffer就是一块内存 xff0c 里面保存着一帧图像 Framebuffer中保存着一帧图像的