Framebuffer应用编程

目录

前言

LCD操作原理

涉及的 API 函数

open函数

ioctl 函数

mmap 函数

Framebuffer程序分析

源码

1.打开设备

2.获取LCD参数

3.映射Framebuffer

4.描点函数

5.随便画几个点

上机实验


前言

本文介绍LCD的操作原理和涉及到的API函数,分析Framebuffer部分代码,


LCD操作原理

在 Linux 系统中通过 Framebuffer 驱动程序来控制 LCD。Frame 是帧的意思,buffer 是缓冲的意思,这意味着 Framebuffer 就是一块内存,里面保存着一帧图像。Framebuffer 中保存着一帧图像的每一个像素的颜色值。Framebuffer的大小为:屏幕分辨率x每一个像素所占的字节数

LCD操作流程如下:
\bullet 驱动程序设置好 LCD 控制器(我并没有学习linux驱动,本文只介绍要用到的函数和分析                 Framebuffer代码,注重应用编程,实在看不懂原理的会用就行,应用为王道)
\bullet APP 使用 ioctl 获得 LCD 分辨率、BPP(每一个像素所占的位数)
\bullet APP 通过 mmap 映射 Framebuffer,在 Framebuffer 中写入数据


(LCD操作原理图)

 问:怎么知道内存中的哪个地址对应哪个元素呢?
        假设需要设置 LCD 中坐标(x,y)处像素的颜色,首要要找到这个像素对应的内存,然后根据它的 BPP 值设置颜色。假设 fb_base 是 APP 执行 mmap 后得到的 Framebuffer 地址,如下图:

假设一个元素的左上角表示这个元素的地址 ,(x,y)坐标的元素前面有y行,在这一行的前面又有x个元素,我们不需要知道它是第几个元素,这样子会很混乱,我们只需要计算在它之前的所有元素占多大内存,那么计算出来的内存空间再加上 fb_base 就是(x,y)坐标元素的地址

可以用以下公式算出(x,y)坐标处像素对应的 Framebuffer 地址:
(x,y)像素起始地址=fb_base+(xres*bpp/8)*y + x*bpp/8

最后一个要解决的问题就是像素的颜色怎么表示?
        
它是用 RGB 三原色(红、绿、 蓝)来表示的,在不同的 BPP 格式中,用不同的位来分别表示 R、G、B。

对于 32BPP,一般只设置其中的低 24 位,高 8 位表示透明度,一般的 LCD 都不支持。
对于 16BPP,常用的是 RGB565;很少的场合会用到 RGB555,这可以通过 ioctl 读取驱动程序中的 RGB 位偏移来确定使用哪一种格式。


涉及的 API 函数

open函数

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>int open(const char *pathname, int flags);
int open(const char *pathname, int flags, mode_t mode);

open函数之前用的很多了,这里就不过多介绍了,可以看我之前的博客Linux文件编程。

ioctl 函数

#include <sys/ioctl.h>int ioctl(int fd, unsigned long request, ...);

函数说明
\bullet fd 表示文件描述符
\bullet request 表示与驱动程序交互的命令,用不同的命令控制驱动程序输出我们需要的数据
\bullet … 表示可变参数 arg;根据 request 命令,设置驱动程序返回输出的数据
\bullet 返回值:打开成功返回文件描述符,失败将返回-1
        ioctl 的作用非常强大、灵活。不同的驱动程序内部会实现不同的 ioctl,APP 可以使用各种 ioctl 跟驱动程序交互:可以传数据给驱动程序,也可以从驱动程序中读出数据。

mmap 函数

#include <sys/mman.h>void *mmap(void *addr, size_t length, int prot, int flags,int fd, off_t offset);

mmap 函数的功能是将文件或其他对象映射到进程的内存地址空间。这使得程序可以通过内存操作直接访问文件内容,而无需使用传统的文件读写函数。其主要功能包括:
\bullet 文件映射:将文件内容映射到内存,使得文件可以像数组一样进行读取和修改
\bullet 共享内存:在多个进程之间共享数据,通过映射同一文件或匿名映射区域实现进程间通信
\bullet 内存映射设备映射设备文件到内存,以便进行高效的数据访问和操作

函数说明
\bullet addr 表示指定映射的內存起始地址,通常设为 NULL 表示让系统自动选定地址,并在成功映射       后返回该地址
\bullet length 表示将文件中多大的内容映射到内存中
\bullet prot 表示映射区域的保护方式,可以为以下 4 种方式的组合
        \circ PROT_EXEC 映射区域可被执行
        \circ PROT_READ 映射区域可被读出
        \circ PROT_WRITE 映射区域可被写入
        \circ PROT_NONE 映射区域不能存取
\bullet Flags 表示影响映射区域的不同特性,常用的有以下两种
        \circ MAP_SHARED 表示对映射区域写入的数据会复制回文件内,原来的文件会改变
        \circ MAP_PRIVATE 表示对映射区域的操作会产生一个映射文件的复制,对此区域的任何修改               都不会写回原来的文件内容中
返回值:若成功映射,将返回指向映射的区域的指针,失败将返回-1


Framebuffer程序分析

源码

#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <linux/fb.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <sys/ioctl.h>static int fd_fb;
static struct fb_var_screeninfo var;	/* Current var */
static int screen_size;
static unsigned char *fb_base;
static unsigned int line_width;
static unsigned int pixel_width;/*********************************************************************** 函数名称: lcd_put_pixel* 功能描述: 在LCD指定位置上输出指定颜色(描点)* 输入参数: x坐标,y坐标,颜色* 输出参数: 无* 返 回 值: 会* 修改日期        版本号     修改人	      修改内容* -----------------------------------------------* 2020/05/12	     V1.0	  zh(angenao)	      创建***********************************************************************/ 
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;	unsigned int red, green, blue;	pen_16 = (unsigned short *)pen_8;pen_32 = (unsigned int *)pen_8;switch (var.bits_per_pixel){case 8:{*pen_8 = color;break;}case 16:{/* 565 */red   = (color >> 16) & 0xff;green = (color >> 8) & 0xff;blue  = (color >> 0) & 0xff;color = ((red >> 3) << 11) | ((green >> 2) << 5) | (blue >> 3);*pen_16 = color;break;}case 32:{*pen_32 = color;break;}default:{printf("can't surport %dbpp\n", var.bits_per_pixel);break;}}
}int main(int argc, char **argv)
{int i;fd_fb = open("/dev/fb0", O_RDWR);if (fd_fb < 0){printf("can't open /dev/fb0\n");return -1;}if (ioctl(fd_fb, FBIOGET_VSCREENINFO, &var)){printf("can't get var\n");return -1;}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;fb_base = (unsigned char *)mmap(NULL , screen_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd_fb, 0);if (fb_base == (unsigned char *)-1){printf("can't mmap\n");return -1;}/* 清屏: 全部设为白色 */memset(fb_base, 0xff, screen_size);/* 随便设置出100个为红色 */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;	
}

1.打开设备

首先打开设备节点:

fd_fb = open("/dev/fb0", O_RDWR);
if (fd_fb < 0)
{printf("can't open /dev/fb0\n");return -1;
}

2.获取LCD参数

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

可以使用以下代码获取 fb_var_screeninfo: 

static struct fb_var_screeninfo var; /* Current var */if (ioctl(fd_fb, FBIOGET_VSCREENINFO, &var))
{printf("can't get var\n");return -1;
}

        注意到 ioctl 里用的参数是:FBIOGET_VSCREENINFO,它表示 get var screeninfo,获得屏幕的可变信息;当然也可以使用 FBIOPUT_VSCREENINFO 来调整这些参数,但是很少用到。

对于固定的参数 fb_fix_screeninfo,在应用编程中很少用到。它的结构体定义如下:

可以使用 ioctl FBIOGET_FSCREENINFO 来读出这些信息,但是很少用到。(F表示fix,确定的,V表示var-variable 可变的)

3.映射Framebuffer

要映射一块内存,需要知道它的地址──这由驱动程序来设置,需要知道它的大小──这由应用程序决定。代码如下:

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;
fb_base = (unsigned char *)mmap(NULL , screen_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd_fb, 0);if (fb_base == (unsigned char *)-1)
{printf("can't mmap\n");return -1;
}

        第3行中,screen_size 是整个 Framebuffer 的大小;PROT_READ | PROT_WRITE 表示该区域可读、可写;MAP_SHARED 表示该区域是共享的,APP 写入数据时,会直达驱动程序,这个参数的更深刻理解可以参考后面驱动基础中讲到的 mmap 知识。(我的理解是将驱动程序所控制的硬件的真实地址映射(mmap)到文件里,linux一切皆文件,然后往地址写数据对硬件进行操作

4.描点函数

        能够在 LCD 上描绘指定像素后,就可以写字、画图,描点函数是基础。代码如下:

1    void lcd_put_pixel(int x, int y, unsigned int color)
2    {
3    	unsigned char *pen_8 = fb_base+y*line_width+x*pixel_width;
4    	unsigned short *pen_16;	
5    	unsigned int *pen_32;	
6
7    	unsigned int red, green, blue;	
8
9	    pen_16 = (unsigned short *)pen_8;
10	    pen_32 = (unsigned int *)pen_8;
11
12	    switch (var.bits_per_pixel)
13      {
14	    	case 8:
15	    	{
16	    		*pen_8 = color;
17	    		break;
18	    	}
19	    	case 16:
20	    	{
21	    		/* 565 */
22	    		red   = (color >> 16) & 0xff;
23	    		green = (color >> 8) & 0xff;
24	    		blue  = (color >> 0) & 0xff;
25		    	color = ((red >> 3) << 11) | ((green >> 2) << 5) | (blue >> 3);
26		    	*pen_16 = color;
27	    		break;
28	    	}
29	    	case 32:
30	    	{
31	    		*pen_32 = color;
32	    		break;
33	    	}
34	    	default:
35	    	{
36	    		printf("can't surport %dbpp\n", var.bits_per_pixel);
37	    		break;
38	    	}
39       }
40    }

        第 1 行中传入的 color 表示颜色,它的格式永远是 0x00RRGGBB,即 RGB888。当 LCD 是 16bpp 时,要把 color 变量中的 R、G、B 抽出来再合并成 RGB565 格式。

        第 3 行计算(x,y)坐标上像素对应的 Framebuffer 地址。

        第 16 行,对于 8bpp,color 就不再表示 RBG 三原色了,这涉及调色板的概念,color 是调色板的值。

        第 22~24 行,先从 color 变量中把 R、G、B 抽出来。

        第 25 行,把 red、green、blue 这三种 8 位颜色值,根据 RGB565 的格式,只保留 red 中的高 5 位、green 中的高 6 位、blue 中的高 5 位,组合成一个新的 16 位颜色值。

        第 26 行,把新的 16 位颜色值写入 Framebuffer。

        第 31 行,对于 32bpp,颜色格式跟 color 参数一致,可以直接写入Framebuffer。

5.随便画几个点

本程序的 main 函数,在最后只是简单地画了几个点:

/* 清屏: 全部设为白色 */
memset(fb_base, 0xff, screen_size);/* 随便设置出100个为红色 */
for (i = 0; i < 100; i++)lcd_put_pixel(var.xres/2+i, var.yres/2, 0xFF0000);

上机实验

板子用的是韦东山老师的imx6ull板,由于我的屏幕被烧坏了,这里就演示不了了。

在 Ubuntu 中编译程序,先设置交叉编译工具链,再执行以下命令:
arm-buildroot-linux-gnueabihf-gcc -o show_pixel show_pixel.c

交叉编译后在板子上就能运行应用程序了,正常会在屏幕中心显示一条红色的线。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1536043.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

进阶岛 renwu5: 茴香豆:企业级知识问答工具实践闯关任务

进阶岛 renwu5: 茴香豆&#xff1a;企业级知识问答工具实践闯关任务 renwu: https://kkgithub.com/InternLM/Tutorial/blob/camp3/docs/L2/Huixiangdou/task.md 在 InternStudio 中利用 Internlm2-7b 搭建标准版茴香豆知识助手&#xff0c;并使用 Gradio 界面完成 2 轮问答&a…

讨论人机交互研究中大语言模型的整合与伦理问题

概述 论文地址&#xff1a;https://arxiv.org/pdf/2403.19876.pdf 近年来&#xff0c;大规模语言模型发展迅速。它们给研究和教育领域带来了许多变化。这些模型也是对人机交互&#xff08;HCI&#xff09;研究过程的有力补充&#xff0c;可以分析定性和定量数据&#xff0c;再…

架构师:在 Spring Cloud 中实现全局异常处理的技术指南

1、简述 在分布式系统中,微服务架构是最流行的设计模式之一。Spring Cloud 提供了各种工具和库来简化微服务的开发和管理。然而,随着服务的增多,处理每个服务中的异常变得尤为复杂。因此,实现统一的全局异常处理成为了关键。本篇博客将介绍如何在 Spring Cloud 微服务架构…

阿里P8和P9级别有何要求

阿里巴巴的P8和P9级别&#xff0c;代表着公司的资深技术专家或管理者岗位&#xff0c;要求候选人具有丰富的职业经历、深厚的技术能力以及出色的领导力。以下是对P8和P9级别的要求、考察点以及准备建议的详细分析。 P8 级别要求 1. 职业经历&#xff1a; 8年以上的工作经验&a…

idea连接数据库大避雷!!!

再跟着黑马学习的时候&#xff0c;用黑马的资料安装的数据库&#xff0c;命令行能正常启动&#xff0c;SQLyog也能正常连接&#xff0c;就是tmd idea连接不了。不论是原始的jdbc,还是其它方式都不行&#xff0c;一直报错&#xff1a; 然后就各种搜&#xff0c;有的说数据库驱动…

SQLite安装(含安装包)

安装包&#xff1a; 通过百度网盘分享的文件&#xff1a;sqlite-dll-win-x64-3460100.zip 链接&#xff1a;https://pan.baidu.com/s/1852coiq51QcNkeaHdu1Oyg 提取码&#xff1a;v2y6 解压 设置环境变量 验证安装成功 SQLite设置完成

恢弘集团SRM采购数字化项目成功上线,企企通助推新材料企业发展新质生产力

近日&#xff0c;企企通携手恢弘集团有限公司&#xff08;以下简称“恢弘集团”&#xff09;打造的一站式数字化采购管理平台正式上线。基于该平台&#xff0c;恢弘集团全流程全周期的数字化采购管理体系进一步升级&#xff0c;在推动企业提高效率的同时&#xff0c;也将形成新…

温习mysql函数 连接查询

字符串 1、CONCAT(S1,S2,...Sn) &#xff1a;字符串拼接&#xff0c;将S1 &#xff0c; S2 &#xff0c; ... Sn 拼接成一个字符串】 2、LOWER(str) &#xff1a;将字符串str全部转为小写 3、UPPER(str) &#xff1a;将字符串str全部转为大写 4、LPAD(str,n,pad)&#xff1a; …

springboot 整合quartz定时任务

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、pom的配置1.加注解 二、使用方法1.工程图2.创建工具类 三、controller 实现 前言 提示&#xff1a;这里可以添加本文要记录的大概内容&#xff1a; 提示&a…

实现拖动标题栏窗口恢复+窗口跟着鼠标移动

窗口跟着鼠标移动 1.重写鼠标按下事件&#xff0c;记录鼠标在窗口中的相对位置 2.重写鼠标移动事件&#xff0c;调用move方法使得窗口移动到鼠标的位置&#xff08;调用globalPos方法获取鼠标的位置&#xff09; 3.注意点&#xff1a;移动时鼠标的位置还要减去一开始的相对位…

一文带你全面了解RAID技术:从基础到进阶的全景解析

一、引言 在如今这个数据爆炸的时代&#xff0c;数据的存储和安全性显得尤为重要。RAID技术作为一种将多块硬盘组合成一个逻辑单元&#xff0c;以实现数据冗余和性能优化的技术&#xff0c;被广泛应用于企业级和个人数据存储中。本文将对RAID的常见级别进行详细解析&#xff0…

[C语言]第九节 函数一基础知识到高级技巧的全景探索

目录 9.1 函数的概念 9.2 库函数 9.2.1 标准库与库函数 示例&#xff1a;常见库函数 9.2.2 标准库与头文件的关系 参考资料和学习工具 如何使用库函数 ​编辑 9.3 ⾃定义函数 9.3.1 函数的语法形式 9.3.2函数的举例 9.4 实参与形参 9.4.1 什么是实参&#xff1f; 9…

影刀RPA实战:网页爬虫之CSDN博文作品数据

今天我们使用影刀来采集网页数据&#xff0c;影刀RPA是一款功能强大的自动化办公软件&#xff0c;它可以模拟人工的各种操作&#xff0c;帮助企业自动处理大量重复性、有逻辑规则的工作。影刀RPA在网页数据采集方面表现出色&#xff0c;能够实现对任何桌面软件、Web程序的自动化…

NeMo Curator 整理用于 LLM 参数高效微调的自定义数据集

目录 概述 预备知识 定义自定义文档构建器 下载数据集 解析和迭代数据集 将数据集写入 JSONL 格式 使用文档构建器加载数据集 使用现有工具统一 Unicode 格式 设计自定义数据集过滤器 编辑所有个人识别信息 添加指令提示 整合管线 概述 出于演示目的&#xff0c;本…

6芯7芯可旋转电连接器航空插头

概述 可旋转电航空插头是一种能够在旋转或相对运动的部件间稳定传输电气信号或电源的装置&#xff0c;广泛应用于航空航天、自动化设备、医疗设备等多个领域。它的核心在于精密的接触系统&#xff0c;由旋转端和固定端两部分组成&#xff0c;通过金属触点或导电环实现电气连接。…

哪些网站用python开发

国内的话&#xff0c;知乎&#xff0c;网易&#xff0c;腾讯&#xff0c;搜狐&#xff0c;金山&#xff0c;豆瓣这些属于用Python比较知名的。大型的项目的话&#xff0c;网易的许多游戏&#xff0c;腾讯的某些网站&#xff0c;搜狐的邮箱&#xff0c;金山的测试框架等等都是或…

实习期间git的分枝管理以及最常用的命令

各位找工作实习的友友在工作之前一定要把git的相关知识掌握呀&#xff0c;我实现期间被leader说过关于git规范的相关问题了 目前已更新系列&#xff1a; 当前&#xff1a;:实习期间git的分枝管理以及最常用的命令 Redis高级-----持久化AOF、RDB原理 Redis高级---面试总结5种…

【JavaEE初阶】多线程(4)

欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 目录 线程安全的 第四个原因 代码举例: 分析原因 解决方法 方法1 方法2 wait(等待)和notify(通知) wait和sleep区别 线程安全的 第四个原因 内存可见性,引起的线程安全问…

springboot3.X版本集成mybatis遇到的问题

由于我本地springboot为3.x版本&#xff0c;如下图所示&#xff0c;最新版本 当我参照如下搜索的内容去集成mybatis的时候&#xff0c;会出现各种各样的报错 最根本的原因是搜出来的配置是参照springboot2.X版本&#xff0c;当我们使用springboot3.x版本之后&#xff0c;需要配…