中文字符的点阵显示

一、指定编码格式

(1)在编写C程序时,可以使用ANSI编码,或是UTF-8编码;一般不使用UTF-16编码,UTF-16编码使用两个字节,其中一个字节是0,在C语言中0表示字符串的结束符,容易引起误会。

(2)在编译程序时,可以使用以下的选项告诉编译器编码方式:

①-finput-charset=GB2312;②-finput-charset=UTF-8

如果不指定“ -finput-charset”, gcc就会默认 C 程序的编码方式为 UTF- 8,即使是以别的格式保存,也会被当作UTF-8来对待。

(3)对于编译出来的可执行程序,可以使用以下的选项编译器指定编码方式:

①-fexec-charset=GB2312;②-fexec-charset=UTF-8

(4)如果“ -finput-charset”与“-fexec-charset”不一样,编译器会进行格式转换。使用gcc及以上选项将GB2312转换为UTF-8:

gcc -o test -finput-charset=GB2312  -fexec-charset=UTF-8 test.c

 二、汉字区位码

HZK16点阵字库:它是常用汉字的16*16的点阵字库。每个汉字用32个字节来描述。采用GB2312来索引编码值。

(1)跟 ASCII 字库一样,每个字节中每一位用来表示一个像素,位值等于 1 时表示对应像素被点亮,位值等于 0 时表示对应像素被熄灭。

(2)HZK16 中是以 GB2312 编码值来查找点阵的,以“中”字为例,它的编码值是“0xd6 0xd0”,其中的 0xd6 表示“区码”,表示在哪一个区:第“0xd6 - 0xa1” 区;其中的 0xd0 表示“位码”,表示它是这个区里的哪一个字符:第“0xd0 - 0xa1”个

(3)每一个区有 94 个汉字。区位码从 0xa1 而不是从 0 开始,是为了兼容 ASCII 码。所以,我们要显示的“中”字,它的 GB2312 编码是 d6d0,它是 HZK16 里第“(0xd6-0xa1)*94+(0xd0-0xa1)”个字符。

三、 使用HZK16显示汉字

3.1 void lcd_put_chinese函数

void lcd_put_chinese(int x, int y, unsigned char *str)
{
    // 计算汉字的区位码中的“区”部分
    unsigned int area  = str[0] - 0xA1;  // 汉字的高字节,减去 0xA1 得到区号(0 开始)
    
    // 计算汉字的区位码中的“位”部分
    unsigned int where = str[1] - 0xA1;  // 汉字的低字节,减去 0xA1 得到位号(0 开始)
    
    // 计算汉字在字库中的实际地址,hzkmem是基地址
    unsigned char *dots = hzkmem + (area * 94 + where) * 32;

    unsigned char byte;  // 用于存储当前行数据的一个字节

    int i, j, b;  // i: 行数, j: 列字节数(每行2字节),b: 当前字节中的位
    for (i = 0; i < 16; i++)  
        for (j = 0; j < 2; j++) 
        {
            // 计算当前行的第 j 个字节数据(每个字节表示 8 个像素)
            byte = dots[i*2 + j];
            
            for (b = 7; b >= 0; b--)
            {
                if (byte & (1 << b))
                {
                    /* 如果当前位为 1,显示白色像素 */
                    lcd_put_pixel(x + j * 8 + 7 - b, y + i, 0xffffff); 
                }
                else
                {
                    /* 如果当前位为 0,显示黑色像素 */
                    lcd_put_pixel(x + j * 8 + 7 - b, y + i, 0);  
                }    
            }
        }
}

x + j * 8 + 7 - b:

j*8 表示当前的字节位置,每行总共两个字节(0,1)

7-b 确认了列的位置

3.2 完整代码

#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>

#define FONTDATAMAX 4096

static const unsigned char fontdata_8x16[FONTDATAMAX] = {

	/* 0 0x00 '^@' */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
	0x00, /* 00000000 */
};

int fd_fb;
struct fb_var_screeninfo var;	/* Current var */
int screen_size;
unsigned char *fbmem;
unsigned int line_width;
unsigned int pixel_width;

int fd_hzk16;
struct stat hzk_stat;
unsigned char *hzkmem;

/**********************************************************************
 * 函数名称: lcd_put_pixel
 * 功能描述: 在LCD指定位置上输出指定颜色(描点)
 * 输入参数: x坐标,y坐标,颜色
 * 输出参数: 无
 * 返 回 值: 会
 ***********************************************************************/ 
void lcd_put_pixel(int x, int y, unsigned int color)
{
	unsigned char *pen_8 = fbmem+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;
		}
	}
}
/**********************************************************************
 * 函数名称: lcd_put_ascii
 * 功能描述: 在LCD指定位置上显示一个8*16的字符
 * 输入参数: x坐标,y坐标,ascii码
 * 输出参数: 无
 * 返 回 值: 无
 ***********************************************************************/ 
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, 0xffffff); /* 白 */
			}
			else
			{
				/* hide */
				lcd_put_pixel(x+7-b, y+i, 0); /* 黑 */
			}
		}
	}
}
/**********************************************************************
 * 函数名称: lcd_put_chinese
 * 功能描述: 在LCD指定位置上显示一个16*16的汉字
 * 输入参数: x坐标,y坐标,ascii码
 * 输出参数: 无
 * 返 回 值: 无
 ***********************************************************************/ 
void lcd_put_chinese(int x, int y, unsigned char *str)
{
	unsigned int area  = str[0] - 0xA1;
	unsigned int where = str[1] - 0xA1;
	unsigned char *dots = hzkmem + (area * 94 + where)*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, 0xffffff); /* 白 */
				}
				else
				{
					/* hide */
					lcd_put_pixel(x+j*8+7-b, y+i, 0); /* 黑 */
				}	
			}
		}
}

int main(int argc, char **argv)
{
	unsigned char str[] = "中";
	
	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;
	fbmem = (unsigned char *)mmap(NULL , screen_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd_fb, 0);
	if (fbmem == (unsigned char *)-1)
	{
		printf("can't mmap\n");
		return -1;
	}

	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;
	}

	/* 清屏: 全部设为黑色 */
	memset(fbmem, 0, screen_size);

    lcd_put_ascii(var.xres/2, var.yres/2, 'A'); /*在屏幕中间显示8*16的字母A*/
	
	printf("chinese code: %02x %02x\n", str[0], str[1]);
	lcd_put_chinese(var.xres/2 + 8,  var.yres/2, str);  //向右移动8个像素

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

3.3 附加lcd_put_str 函数

作用:可以输出混合的中英文字符,比如“中国china”


/**********************************************************************
 * 函数名称: lcd_put_str
 * 功能描述: 在LCD指定位置上显示汉字与字母结合的字符串,且能自动换行
 * 输入参数: x坐标,y坐标,str字符串首地址
 * 返 回 值: 无
 ***********************************************************************/ 
void lcd_put_str(int x, int y, unsigned char *str)
{
	int x_offset = 0;
	for(int i = 0; str[i] != '\0'; i++)
	{
		//若超过LCD边界则换行
		if(x + x_offset > var.xres - 16)
		{
			x = 0;
			x_offset = 0;
			y += 16;
		}
		//分辨ascii还是中文
		if(str[i] > 0 && str[i] <= 0x7f)//str[i]为ascii码内的值
		{
			lcd_put_ascii(x + x_offset, y, str[i]);
			x_offset += 8;
		}
		else//中文
		{
			lcd_put_chinese(x + x_offset, y, &str[i]);
			x_offset += 16;
            i++;
		}
	}
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值