Linux 下 使用点阵在LCD上显示汉字,字符

本文详细讲解如何在Linux的LCD设备上使用点阵显示汉字和字符,涉及字符点阵获取、Framebuffer操作、汉字库的使用等。通过获取字符点阵、描点函数、打开LCD设备、ioctl获取参数、mmap映射内存以及清屏显示等功能实现字符和汉字的显示。同时介绍了汉字的区位码和HZK16汉字库的使用方法,提供显示汉字的函数。文章适合对Linux底层编程和LCD显示感兴趣的读者。
摘要由CSDN通过智能技术生成


前言

这篇文章主要讲一下如何在 LCD 上使用点阵显示汉字,字符 ,修改颜色 及效果展示。其中包含了几个核心函数,我们需要了解。

一、显示字符

1.获取点阵:

各个字符对应的点阵都保存在一个数组里,大家可以打开 font_8x16.c 中得到点阵。(不同的点阵对应不同的代码,这里我使用的是 8x16 的点阵 )

	unsigned char *dots = (unsigned char *)&fontdata_8x16[c*16];

2.描点(显示字符函数):

找到点阵,将里面需要的点上色就可以在 LCD 上显示出不同的字符
我们使用的是 8x16 的点阵 ,点阵有十六行,每一行里有8位。所以首先要有一个循环16次的大循环代表每一行,在每一个大循环里也需要一个循环8次的小循环,代表每一位。

x ,y : 代表对应点的横纵坐标
c : 表示你要显示的字符
lcd_put_pixel 此函数为描点函数,在 Linux 应用基础 Framebuffer应用编程 中我已详细的写出该函数。(第一二个参数代表x,y 坐标 。第三个参数代表颜色,可任意修改。)

void lcd_put_ascii(int x, int y, unsigned char c)
{
	unsigned char *dots = (unsigned char *)&fontdata_8x16[c*16];
	int i, b;
	unsigned char byte;

	for (i = 0; i < 16; i++)
	{
		byte = dots[i];
		for (b = 7; b >= 0; b--)
		{
			if (byte & (1<<b))
			{
				/* show */
				lcd_put_pixel(x+7-b, y+i, 0x00ff00); /* 绿色 */
			}
			else
			{
				/* hide */
				lcd_put_pixel(x+7-b, y+i, 0); /* 黑 */
			}
		}
	}
}

3. 要打开LCD设备:

对于不同的板子,你的 framebuffer 节点可能有所不同,需注意。

	int fd_fb;
	
	fd_fb = open("/dev/fb0", O_RDWR);

4. 通过ioctl 获取Framebuffer参数:

将得到的数据保存在 var 结构体中。这个结构体如果不了解的可以参考 Linux 应用基础 Framebuffer应用编程,我都已详细的说明。

	struct fb_var_screeninfo var;
	
    if (ioctl(fd_fb, FBIOGET_VSCREENINFO, &var))

5. 通过mmap映射出Framebuffer的地址:

要映射一块内存,就需要知道它的大小和地址。
screen_size : 为该内存的地址。(整个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;
	fbmem = (unsigned char *)mmap(NULL , screen_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd_fb, 0);

6.清屏并显示字符:

使用 memset 将屏幕变成黑色,再显示字符,便于观察。

	memset(fbmem, 0, screen_size);

	lcd_put_ascii(var.xres/2, var.yres/2, 'A'); /*在屏幕中间显示8*16的字母A*/

注意: 上面我们映射了一块内存,我们使用完后需要释放 munmap

munmap(fbmem , screen_size);

二、显示汉字

显示汉字,我们可以从网上下载 HZK16 这个文件, 它是常用汉字的 16*16 点阵字库。HZK16里每个汉字使用32字节来描述

1.区位码:

HZK16中是以 GB2312 编码值来查找点阵的,这里就要涉及汉字的区位码。区位码把GB2312字符集分为94个区,每区含有94个位。一个汉字所在的区号和位号简单地组合在一起就构成了该汉字的"区位码"。

例如:
以“中”字为例,它的编码值是“0xd6 0xd0”,其中的0xd6表示“区码”,表示在哪一个区。其中的0xd0表示“位码”,表示它是这个区里的哪一个字符。由于区位码从0xA1开始,所以 “中” 字在 第“0xd6 - 0xa1”区,第“0xd0 - 0xa1”个。

	unsigned int area  = str[0] - 0xA1; //区码
	unsigned int where = str[1] - 0xA1; //位码
	unsigned char *dots = hzkmem + (area * 94 + where)*32;//每一个区有94个汉字每个汉字占了32个字节

2. 打开汉字库文件:

struct stat 这个结构体是用来描述一个linux系统文件系统中的文件属性的结构。通过 fstat 函数获得文件状态,保存到 hzk_stat 结构体中。

	int fd_hzk16;
	struct stat hzk_stat;

	fd_hzk16 = open("HZK16", O_RDONLY);
	if (fd_hzk16 < 0)
	{
		printf("can't open HZK16\n");
		return -1;
	}
	if(fstat(fd_hzk16, &hzk_stat))
	{
		printf("can't get fstat\n");
		return -1;
	}
	hzkmem = (unsigned char *)mmap(NULL , hzk_stat.st_size, PROT_READ, MAP_SHARED, fd_hzk16, 0);
	if (hzkmem == (unsigned char *)-1)
	{
		printf("can't mmap for hzk16\n");
		return -1;
	}

下面是 struct stat 结构体:

struct stat {
		mode_t     st_mode;       //文件对应的模式,文件,目录等
	    ino_t      st_ino;       //inode节点号
		dev_t      st_dev;        //设备号码
		dev_t      st_rdev;       //特殊设备号码
		nlink_t    st_nlink;      //文件的连接数
		uid_t      st_uid;        //文件所有者
		gid_t      st_gid;        //文件所有者对应的组
		off_t      st_size;       //普通文件,对应的文件字节数
		time_t     st_atime;      //文件最后被访问的时间
		time_t     st_mtime;      //文件内容最后被修改的时间
		time_t     st_ctime;      //文件状态改变时间
		blksize_t st_blksize;    //文件内容对应的块大小
		blkcnt_t   st_blocks;     //伟建内容对应的块数量
	 };

3.显示汉字函数:

void lcd_put_chinese(int x, int y, unsigned char *str,unsigned int color)
{
	unsigned int area  = str[0] - 0xA1; //区码
	unsigned int where = str[1] - 0xA1; //位码
	unsigned char *dots = hzkmem + (area * 94 + where)*32;//每一个区有94个汉字每个汉字占了32个字节
	unsigned char byte;

	int i, j, b;
	for (i = 0; i < 16; i++)
	{
		for (j = 0; j < 2; j++)
		{
			byte = dots[i*2 + j];
			for (b = 7; b >=0; b--)
			{
				if (byte & (1<<b))
				{
					/* show */
					lcd_put_pixel(x+j*8+7-b, y+i, 0x00ff00); /* 绿色 */
				}
				else
				{
					/* hide */
					lcd_put_pixel(x+j*8+7-b, y+i, 0); /* 黑 */
				}	
			}
		}
	}
}

可在主函数里面调用 :

	unsigned char str1[] = "你";
	unsigned char str2[] = "好";

	lcd_put_chinese(var.xres/2 + 8,  var.yres/2, str1);
	lcd_put_chinese(var.xres/2 + 8+16,  var.yres/2, str2);


总结

效果:
在这里插入图片描述

大家可以自行实践一下,有不懂的欢迎留言讨论。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

糖果罐子♡

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值