中文字符的点阵显示

337 阅读2分钟

指定编码格式

image.png

-finput-charset=GB2312
-finput-charset=UTF-8
-fexec-charset=GB2312
-fexec-charset=UTF-8

GB2312 转为 UTF-8

image.png

UTF-8 转为 GB2312

image.png

汉字区位码

我们从网上搜到 HZK16 这个文件,它是常用汉字的 16*16 点阵字库。HZK16里每个汉字使用 32 字节来描述,如图所示: image.png

打开汉字库文件

image.png

image.png

程序代码

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

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_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); /* 黑 */
        }
      }
    }
}

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;
  }
  // 读取HZK16文件
  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);

  printf("chinese code: %02x %02x\n", str[0], str[1]);
  lcd_put_chinese(var.xres / 2 + 8, var.yres / 2, str);

  munmap(fbmem, screen_size);
  close(fd_fb);

  return 0;
}

编译程序

arm-buildroot-linux-gnueabihf-gcc -fexec-charset=GB2312 -o show_chinese show_chinese.c

image.png