0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Linux应用开发-LCD显示BMP图片

DS小龙哥-嵌入式技术 来源:DS小龙哥-嵌入式技术 作者:DS小龙哥-嵌入式技 2022-08-14 09:18 次阅读

1. 前言

BMP是一种与硬件设备无关的图像文件格式,是Windows环境中交换与图有关的数据的一种标准,在Windows环境中运行的图形图像软件都支持BMP图像格式。BMP格式的图片存放的就是原始的RGB数据,一般没有做压缩,也就是图片的画质是最原始的,也导致BMP图片占用的内存非常大。现在常用的jpg、jpeg格式都是压缩格式,保存的时候通过算法编码压缩,显示的时候再解压成RGB数据渲染显示。

BMP格式在嵌入式设备里用的还是较多的,BMP虽然占用内存大,优点是显示速度快,因为不需要解码,在性能一般,不是很强的设备上使用BMP显示效率较高。

为了解BMP格式,这篇文章就采用Linux开发板作为实验平台,在LCD屏上读取BMP图片,完成绘制,不需要借助任何第三方库,全部由纯C语言代码一行一行敲出来,深入理解Linux下帧缓冲编程框架、BMP图片的存储结构原理。

一般BMP图片由以下4个部分组成: 1:文件头 2:图像参数 3:调色板 4:位图数据

现在一般采用的图片都是RGB888,24位真彩色,就没有调色板,只有3个部分组成。

其中文件头存放图片的属性,位图数据偏移量。图像参数存放图片的宽高、像素位数等信息。位图数据就是存储的原始RGB数据,可以直接在LCD屏上显示。

下面列出BMP图片的结构:

image-20220124001914847image-20220124001941914

位图数据存储规则:

(1)每行的字节数必须是4的倍数,如果不是,则需要用0补齐。 (2)BMP位图数据的存放是从下到上,从左到右的。先读最后一行,读完后在读倒数第二行。

image-20220124002345869

按照上面的介绍,就可以定义一个BMP解码专用的结构体,对应文件里每个字节数据,结构体成员变量必须按照上面截图里的说明定义。整个结构体还需要进行强制1个字节对齐,不然每个编译器对结构体的空间开辟规则有差异,会导致数据错位。

#pragma pack(1) //强制1个字节对齐
//BMP的文件头
struct _BMP_HEAD
{
    char type[2]; //图片的类型 "BM"
    unsigned int size; //文件大小
    unsigned short  r1; //保留1
    unsigned short  r2; //保留2
    unsigned int seek; //数据偏移字节(真实像素点数据)
};

//BMP的参数信息
struct _BMP_INFO
{
    unsigned int size; //当前结构体大小
    unsigned int w; //宽度
    unsigned int h; //高度
    unsigned short flag; //固定为1
    unsigned short bit; //像素点的位数
    unsigned int r1; //压缩方式  0
    unsigned int r2; //水平分辨率
    unsigned int r3; //垂直分辨率
    unsigned int r4; //垂直分辨率
    unsigned int r5; //引用色彩
    unsigned int r6; //关键色彩
};
复制代码

2. 实现代码

要在LCD屏上完成BMP图片的显示,编写代码需要分几步完成,先编写LCD屏的基本显示代码,封装画点函数,LCD屏测试没有问题之后,再编写BMP解码代码,完成图片的渲染显示。

2.1 封装LCD屏画点函数

#include 
#include 
#include 
#include 
unsigned char *fb_mem;
struct fb_var_screeninfo var;//可变参数
struct fb_fix_screeninfo fix;//固定参数
​
​
/*画点*/
void show_pixel(int x,int y,int color)
{
    unsigned long *show32 = NULL;
    /* 定位到LCD屏上的位置*/
    show32=(unsigned long*)(fb_mem+y*var.xres*var.bits_per_pixel/8 + x*var.bits_per_pixel/8); 
 *show32 =color; /*向指向的LCD地址赋数据*/
}
​
​
int main(int argc,char**argv)
{
​
     int fb;
     fb=open("/dev/fb0",2);
     if(fb<0)
         {
            printf("fb0打开失败!\n");
            return -1;   
         }
         
      /*1. 获取可变参数*/
      ioctl(fb,FBIOGET_VSCREENINFO,&var);
      printf("x=%d\n",var.xres);
      printf("y=%d\n",var.yres);
      printf("bit=%d\n",var.bits_per_pixel);
      
      /*2. 获取固定参数*/
      ioctl(fb,FBIOGET_FSCREENINFO,&fix);
      printf("line_byte=%d\n",fix.line_length);
      printf("smem_len=%d\n",fix.smem_len);
      
      /*3. 映射LCD地址*/
      fb_mem=mmap(NULL,fix.smem_len,PROT_READ|PROT_WRITE,MAP_SHARED,fb,0);
      
      int i,j;
      for(i=0;i;i++)>

2.2 显示BMP图片

在工程目录下准备几张测试的BMP图片,程序运行时,在命令行上传入要显示的图片文件地址接口

image-20220124002843632
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#pragma pack(push) /* 将当前pack设置压栈保存 */
#pragma pack(1)    /* 必须在结构体定义之前使用,这是为了让结构体中各成员按1字节对齐 */

/* 1、需要文件信息头:14个字节 */
typedef struct tagBITMAPFILEHEADER { /* bmfh */
	unsigned short bfType;      //保存图片类似。 'BM'
	unsigned long  bfSize;      //图片的大小
	unsigned short bfReserved1;
	unsigned short bfReserved2; 
	unsigned long  bfOffBits;  //RGB数据偏移地址
}BITMAPFILEHEADER;

/* 位图信息头 */
typedef struct tagBITMAPINFOHEADER { /* bmih */
	unsigned long  biSize;      //结构体大小
	unsigned long  biWidth;		//宽度
	unsigned long  biHeight;	//高度
	unsigned short biPlanes;
	unsigned short biBitCount;	//颜色位数
	unsigned long  biCompression;
	unsigned long  biSizeImage;
	unsigned long  biXPelsPerMeter;
	unsigned long  biYPelsPerMeter;
	unsigned long  biClrUsed;
	unsigned long  biClrImportant;
} BITMAPINFOHEADER;
#pragma pack(pop) /* 恢复先前的pack设置 */


unsigned char *fbmem=NULL;
struct fb_var_screeninfo var;
struct fb_fix_screeninfo fix;
static int iFileSize = 0;




void show_pixel(int x,int y,int color)
{
	unsigned char *show8=NULL;
	unsigned short *show16=NULL;
	unsigned long *show32 = NULL;
	int red;
	int green;
	int blue;
	/* 定位到LCD屏上的位置 */
	show8 = fbmem + y*var.xres*var.bits_per_pixel/8 + x*var.bits_per_pixel/8;
	show16 = (unsigned short *)show8;
	show32 = (unsigned long *)show8;
	switch(var.bits_per_pixel)
	{
		case 8:

		{
			*show8 = color;
			break;
		}
		case 16:
		{
			/* RGB:565 */
			red = (color >> 16)&0xff;
			green = (color >> 8)&0xff;
			blue  = color&0xff;
			color = ((red>>3)<<11) | ((green>>2)<<6) |(blue>>3);
			*show16 = color;
			break;
		}
		case 32:
		{
			*show32 = color;
			break;
		}
		default:break;
	}
	
	
}

/*映射图片地址*/
static unsigned char *getbmpadd(char *name)
{
	unsigned char *bmpmem = NULL;
	FILE* filp;
	int fd;
	struct stat t_stat;
	/* 以r+可读可写方式打开name */
	filp = fopen(name,"r+");
	if(filp == NULL)
	{
		printf("can't open %s\n",name);
		return NULL;
	}
	/* 把文件指针转化为文件描述符 */
	fd = fileno(filp);
	/* 获取文件大小 */
	fstat(fd, &t_stat);
	iFileSize = t_stat.st_size;
	/* 映射 */
	bmpmem = mmap(NULL,iFileSize,PROT_READ|PROT_WRITE,MAP_SHARED,fd, 0);
	if(bmpmem == (unsigned char *)-1)
	{
		printf("can't mmap..\n");
		return NULL;
	}
	return bmpmem;
}
void bmp_destroy(unsigned char *bmpmem)
{
	munmap(bmpmem,iFileSize);
}

void Convert_One_Line(unsigned char *src,unsigned char *dst,int iWidth)
{
	unsigned char *show8=NULL;
	unsigned short *show16=NULL;
	unsigned long *show32 = NULL;
	unsigned char *buf = src;
	int red;
	int green;
	int blue;
	int i;
	/* 定位到LCD屏上的位置 */
	show8 = dst;
	show16 = (unsigned short *)show8;
	show32 = (unsigned long *)show8;
	
	for(i=0;i>3)<<11) | ((green>>2)<<6) |(blue>>3);
				show16++;
				break;
			}
			case 32:
			{
				*show32 = (red<<16)|(green<<8)|blue;
				show32++; // 4个字节
				break;
			}
			default:break;
		}
	}
}


/* 获取颜色阵列数据 */
int getbmpandshow(unsigned char *bmpmem)
{
	/* 定义文件信息头 */
	BITMAPFILEHEADER *bithead;
	/* 定义文件参数信息 */
	BITMAPINFOHEADER *bitinfo;
	

	unsigned char *src=NULL;
	unsigned char *dst=NULL;
	int iWidth;
	int iHeight;
	int iBpp;
	int iLineWidth;
	int iRealLineWidth;
	int iFbLineWidth;
	int i=0;
	
	/* 获取文件信息头起始地址 */
	bithead =(BITMAPFILEHEADER *)bmpmem;
	/* 获取位图信息头起始地址 */
	bitinfo = (BITMAPINFOHEADER *)(bmpmem + sizeof(BITMAPFILEHEADER));
	
	iWidth  = bitinfo->biWidth;
	iHeight = bitinfo->biHeight;
	iBpp  	= bitinfo->biBitCount;
	printf("iWidth = %d\n",iWidth);
	printf("iHeight = %d\n",iHeight);
	printf("iBpp = %d\n",iBpp);
	
	/* 找到颜色阵列,RGB数据的起始地址 */
	src = bmpmem + bithead->bfOffBits;
	
	/* 得到图片一行字节数 */
	iLineWidth = iWidth*iBpp/8; 

	/* 向4取整,保证一行必须是4的倍数 */ 
	iRealLineWidth = (iLineWidth+3)&~0x3;  //  iLineWidth % 4  =0

	/* src指向图片RGB数据最后一行的首地址*/
	src += iRealLineWidth*(iHeight-1);

	/* LCD屏一行的总字节数  */
	iFbLineWidth = var.xres * var.bits_per_pixel/8;

    /*dst指向LCD的首地址*/
	dst = fbmem;

	for(i=0;i\n",argv[0]);
		return -1;
	}
	lcd_init(argv[1]);

	/* 4、清屏 */
	memset(fbmem,0x0,fix.smem_len);
	
	/* 4.1、显示图片-映射图片地址 */
	bmpmem = getbmpadd(argv[2]); 
	if(NULL == bmpmem)
	{
		printf("can't get bmp address!!\n");
		return -1;
	}
	getbmpandshow(bmpmem); // 显示图片
	bmp_destroy(bmpmem); //释放映射的空间
	return 0;
};i++)>;i++)>

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • lcd
    lcd
    +关注

    关注

    34

    文章

    4426

    浏览量

    167461
  • Linux
    +关注

    关注

    87

    文章

    11303

    浏览量

    209439
  • BMP
    BMP
    +关注

    关注

    0

    文章

    48

    浏览量

    17062
收藏 人收藏

    评论

    相关推荐

    飞凌嵌入式ElfBoard ELF 1板卡-LCD显示图片编程示例之开发板测试

    :~# ./elf1_cmd_lcd_bmp fb0 -b 16 -n logo.bmp logo.bmp.xres=800, .yres=480, .bit=16 .xres_virtual=800, .yres_virt
    发表于 11-25 09:36

    移植SDL到JZ2440显示BMP图片

    LOGO的制作, 韦老师第3期讲了如何显示jpeg图片, 那么怎么显示bmp图片?这次我们借助libSDL来实现,我们先移植SDL到Ubun
    发表于 06-29 11:19

    ILI9325屏幕显示bmp图片出现分段

    目前在做从SD卡解析bmp图片显示到TFT上的功能。图片已经能显示出来了。但是出现如图中图片分段
    发表于 01-14 08:57

    请问解码BMP图片程序在哪?

    ;BMPFile,x,y,member,denom);}break; case 1://在LCD中间显示图片XSize = GUI_BMP_GetXSizeEx(BmpGetData,
    发表于 09-24 01:57

    bmp图片LCD开发板上显示的方法

    bmp图片原理,开头54字节需要去掉。bmp是3字节的RGB,而且是内存是翻转过来的,就是说bmp第一行内存是lcd的最后一行才可以正常
    发表于 12-27 07:41

    FSMC_液晶显示BMP图片.pdf

    FSMC 显示BMP 图片,感兴趣的可以看看。
    发表于 01-18 14:16 7次下载

    基于STM32芯片的LCD显示屏的BMP图片显示

    基于STM32芯片的LCD显示屏的BMP图片显示
    发表于 01-18 14:39 61次下载

    LCD彩色图片转换工具BMP_to_H

    LCD彩色图片转换工具BMP_to_H
    发表于 12-28 10:16 13次下载

    嵌入式linux操作framebuffer显示bmp图片

    编译后拷贝进开发板即可使用使用方法./fb_show_bmp test.bmp显示图片由参数指定,上面指令中test.
    发表于 11-01 17:06 10次下载
    嵌入式<b class='flag-5'>linux</b>操作framebuffer<b class='flag-5'>显示</b><b class='flag-5'>bmp</b><b class='flag-5'>图片</b>

    嵌入式LinuxLCD应用编程: 调用giflib库解码显示GIF动态图

    一、开发环境介绍开发板:友善之臂Tiny4412LCD型号:S702 、分辨率: 800*480Linux内核版本:Linux 3.5交叉编
    发表于 11-02 13:21 5次下载
    嵌入式<b class='flag-5'>Linux</b>下<b class='flag-5'>LCD</b>应用编程: 调用giflib库解码<b class='flag-5'>显示</b>GIF动态图

    剖析BMP图片文件

    剖析BMP图片文件
    发表于 01-23 10:05 2次下载

    Linux开发_文件目录操作介绍、创建BMP图片

    介绍C语言文件编程,创建BMP图片BMP图片结构介绍,介绍目录接口编程,Makefile知识点。
    的头像 发表于 09-17 15:44 1490次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>开发</b>_文件目录操作介绍、创建<b class='flag-5'>BMP</b><b class='flag-5'>图片</b>

    Linux开发_介绍BMP图片上下翻转、添加水印

    介绍BMP图片上下翻转、添加水印、通过学习BMP图片结构学习文件编程。
    的头像 发表于 09-17 15:43 1878次阅读

    Linux开发_BMP图片编程(翻转、添加水印)

    接收BMP图片编程,完成BMP图片翻转、水印添加等知识点。
    的头像 发表于 09-17 15:33 1552次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>开发</b>_<b class='flag-5'>BMP</b><b class='flag-5'>图片</b>编程(翻转、添加水印)

    国产DSP操作教程:LCD图片显示实验(程序读取图片

    学习BMP图像数据的存储格式,并实现LCD图像数据的读取与显示
    的头像 发表于 10-19 12:33 2017次阅读
    国产DSP操作教程:<b class='flag-5'>LCD</b><b class='flag-5'>图片</b><b class='flag-5'>显示</b>实验(程序读取<b class='flag-5'>图片</b>)