LCD工作时会在内存中的显存(framebuffer)中取出若干个字节来表示一个像素点,如此循环的把显存中的点阵显示出来
下面直接贴出代码,在代码中有详细的注释:
int main(int argc,char **argv)
{unsigned char str[] = "中"; //1.打开文fb0文件//2.先获取LCD的相关信息,如分辨率等,可变信息和固定信息fd_fb = open("/dev/fb0", O_RDWR); //打开lcd设备fb0if (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;}}
}