思路:
主要是判断中文和非中文字符的条件是什么?从《嵌入式Linux应用开发完全手册V4.0_韦东山全系列视频文档-IMX6ULL开发板.docx》手册中我们可以看到以下信息:
HZK16中是以GB2312编码值来查找点阵的,以“中”字为例,它的编码值是“0xd6 0xd0”,其中的0xd6表示“区码”,表示在哪一个区:第“0xd6 - 0xa1”区;其中的0xd0表示“位码”,表示它是这个区里的哪一个字符:第“0xd0 - 0xa1”个。每一个区有94个汉字。区位码从0xa1而不是从0开始,是为了兼容ASCII码。
从上面这一段我们可以得知汉字是以区位码(区码+位码)的形式在字库中存放的,区位码是从0xa1开始的,而英文的ASCII码对应的十六进制区间为[0x41~0x7A],故我们可以以0xa1为判断条件来判断输入的字符串中哪些是中文,哪些是非中文。
具体函数实现如下:
/**********************************************************************
* 函数名称: lcd_put_str
* 功能描述: 在LCD指定位置可以显示混合的中英文字符,比如“中国china”,支持自动换行
* 输入参数: x坐标,y坐标,字符串str
* 输出参数: 无
* 返 回 值: 无
* 修改日期 版本号 修改人 修改内容
* -----------------------------------------------
* 2021/09/27 V1.0 Cwx 创建
***********************************************************************/
void lcd_put_str(int x, int y, unsigned char *str, unsigned int show_color)
{
unsigned int index = 0;
while (*str)
{
if (*str <= 0xa1)
{
if (*str == '\r' || *str == '\n')
{
y += 16;
index = 0;
}
else
{
lcd_put_ascii(x + index, y, *str, show_color);
index += 8;
}
str++;
}
else
{
lcd_put_chinese(x + index, y, str, show_color);
index += 16;
str+=2;
}
}
}
完整的演示代码如下:
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坐标,颜色
* 输出参数: 无
* 返 回 值: 会
* 修改日期 版本号 修改人 修改内容
* -----------------------------------------------
* 2020/05/12 V1.0 zh(angenao) 创建
***********************************************************************/
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码
* 输出参数: 无
* 返 回 值: 无
* 修改日期 版本号 修改人 修改内容
* -----------------------------------------------
* 2020/05/12 V1.0 zh(angenao) 创建
***********************************************************************/
void lcd_put_ascii(int x, int y, unsigned char c, unsigned int show_color)
{
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, show_color); /* 白 */
}
else
{
/* hide */
lcd_put_pixel(x+7-b, y+i, 0); /* 黑 */
}
}
}
}
/**********************************************************************
* 函数名称: lcd_put_chinese
* 功能描述: 在LCD指定位置上显示一个16*16的汉字
* 输入参数: x坐标,y坐标,ascii码
* 输出参数: 无
* 返 回 值: 无
* 修改日期 版本号 修改人 修改内容
* -----------------------------------------------
* 2020/05/12 V1.0 zh(angenao) 创建
**********************************************************************
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); // 黑
}
}
}
}
*/
/**********************************************************************
* 函数名称: lcd_put_chinese
* 功能描述: 在LCD指定位置上显示一个16*16的汉字
* 输入参数: x坐标,y坐标,汉字字符
* 输出参数: 无
* 返 回 值: 无
* 修改日期 版本号 修改人 修改内容
* -----------------------------------------------
* 2020/05/12 V1.0 zh(angenao) 创建
***********************************************************************/
void lcd_put_chinese(int x, int y, unsigned char *str, unsigned int show_color)
{
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, show_color); // 白
}
else
{
// hide
lcd_put_pixel(x+j*8+7-b, y+i, 0); // 黑
}
}
}
}
/**********************************************************************
* 函数名称: lcd_put_str
* 功能描述: 在LCD指定位置可以显示混合的中英文字符,比如“中国china”,支持自动换行
* 输入参数: x坐标,y坐标,字符串str
* 输出参数: 无
* 返 回 值: 无
* 修改日期 版本号 修改人 修改内容
* -----------------------------------------------
* 2021/09/27 V1.0 Cwx 创建
***********************************************************************/
void lcd_put_str(int x, int y, unsigned char *str, unsigned int show_color)
{
unsigned int index = 0;
while (*str)
{
if (*str <= 0xa1)
{
if (*str == '\r' || *str == '\n')
{
y += 16;
index = 0;
}
else
{
lcd_put_ascii(x + index, y, *str, show_color);
index += 8;
}
str++;
}
else
{
lcd_put_chinese(x + index, y, str, show_color);
index += 16;
str+=2;
}
}
}
int main(int argc, char **argv)
{
unsigned char str[] = "中国china";
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_str(var.xres/2 + 8, var.yres/2, str, 0xFF0000);
munmap(fbmem , screen_size);
close(fd_fb);
return 0;
}
接下来大家应该比较熟悉了,编译 - 拷贝 - 执行!
祝大家学习顺利,调试愉快......