LCD工作时会在内存中的显存(framebuffer)中取出若干个字节来表示一个像素点,如此循环的把显存中的点阵显示出来
下面直接贴出代码,在代码中有详细的注释:
int main(int argc,char **argv)
{
unsigned char str[] = "中";
//1.打开文fb0文件
//2.先获取LCD的相关信息,如分辨率等,可变信息和固定信息
fd_fb = open("/dev/fb0", O_RDWR); //打开lcd设备fb0
if (fd_fb < 0)
{
printf("can't open /dev/fb0\n");
return -1;
}
if (ioctl(fd_fb, FBIOGET_VSCREENINFO, &var)) //获取lcd设备fb0可变信息
{
printf("can't get var\n");
return -1;
}
if (ioctl(fd_fb, FBIOGET_FSCREENINFO, &fix)) // 获取lcd设备fb0固定信息
{
printf("can't get fix\n");
return -1;
}
line_width = var.xres * var.bits_per_pixel / 8; //算出一行的宽度单位是字节
pixel_width = var.bits_per_pixel / 8; //算出每个像素的字节数
//算出显存的大小var.xres屏幕宽度 var.yres屏幕高度 var.bits_per_pixel每个像素用几字节表示
screen_size = var.xres * var.yres * var.bits_per_pixel / 8;
//给lcd设备分配一个screen_size大小的显存空间
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_RDWR); //打开汉字库
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'); //显示一个英文字母
printf("chinese code : %02x %02x \n",str[0],str[1]);
lcd_put_chinese(var.xres/2 + 8, var.yres/2, str); //显示一个中文
return 0;
}
显示字符函数
void lcd_put_ascii(int x, int y, unsigned char c)
{
//fontdata_8x16时字符集数组的地址,而每个字符是8*16的大小也就是16字节,
//字符c的ASCII值就代表了它在数组中的位置,如字符'A',的ASCII值是0x41=65
//它在数组中的位置就是65个,
unsigned char *dots = ( unsigned char *)&fontdata_8x16[c*16];
int i,b;
unsigned char byte;
for(i = 0; i < 16; i++) //16行数据
{
byte = dots[i]; //每次取出一个字节
for(b = 7; b >= 0; b--)
{
if(byte &(1<<b)) //比较每一位的值,1代表要显示 0 代表不显示
{
//显示像素
lcd_put_pixel(x+7-b,y+i,0xffffff);//白色
}
else
{
//不显示像素
lcd_put_pixel(x+7-b,y+i,0);//黑色
}
}
}
}
显示汉字函数
void lcd_put_chinese(int x, int y,unsigned char *str)//国标码两个字节
{
unsigned int arse = str[0] - 0xA1; //算出区码
unsigned int where = str[1] - 0xA1; //算出区码中的偏移位置
unsigned char *dots = hzkmem + (arse*94 + where) * 32;//算出要显示的汉字在字库文件中的位置
int i,j,b;
unsigned char byte;
for(i = 0; i<16; i++) //16行
{
for(j = 0; j < 2; j++) //一行两个字节 用j把它分开
{
byte = dots[i*2+j]; //算出要打印的数据位置
for(b = 7; b>=0; b--)
{
if(byte & (1<<b))
{
lcd_put_pixel(x+j*8+7-b, y+i, 0xffffff);
}
else
{
lcd_put_pixel(x+j*8+7-b, y+i, 0);
}
}
}
}
}
在GBK码中的字符“中”字,它的GBK码是D6 D0,其中D6是区码,D0是位码,每一个区有94个字符,每一个字符是16*16位,也就是32字节,因此unsigned char dots = hzkmem + (arse94 + where) * 32;//表示算出要显示的汉字在字库文件中的位置
显示像素函数
void lcd_put_pixel(int x, int y, unsigned int color)
{
//对于不同的像素宽度,处理方式不同
//参数中颜色值color是一个int型的变量,它表示的颜色的格式是RGB格式
//0x00RRGGBB,只用到了三个字节表示颜色
unsigned char *pen_8 =fbmem + y*line_width + x*pixel_width;
unsigned short *pen_16 = (unsigned short*)pen_8;
unsigned int *pen_32 = (unsigned int * )pen_8;
unsigned int red,green,blue;
switch(var.bits_per_pixel)
{
case 8: //
{
*pen_8 = color;
break;
}
case 16: //对于像素宽度位16的,需要把int型的数据转化为565的16位数据
{ //之后再重新组装成16位的数据,显示出来
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:
{ //对于32位的数据,可以直接显示
*pen_32 = color;
break;
}
default :
{
printf("can't surport %dbpp\n", var.bits_per_pixel);
break;
}
}
}