s3c2440的LCD字符显示

发布者:epsilon15最新更新时间:2016-05-30 来源: eefocus关键字:s3c2440  LCD  字符显示 手机看文章 扫描二维码
随时随地手机看文章
在上一篇文章中,我们详细介绍了如何驱动LCD,对于在LCD屏上呈现各种简单的图形已经不是一件高不可攀的难事。但如何绘制字符呢?
 
其实每一字符就是一幅图像,字符的大小对应于图像的大小,字符的笔画对应于图像的内容。那么如何把字符转换为图像呢?简单的方法是使用“字模提取”之类的软件,它能够把任意的字符转换为一个字节型的数组,数组元素中的每一位代表LCD上的一个像素点,当为1时,表示该位置为字符的一个笔画,需要上色,而为0时,表示不是笔画,不需要上色。例如,一个字符想要在16×16的面积上显示,即该字符的宽和高各为16个像素,因为每一个像素用一位来表示,因此用字模提取软件生成的字节型数组,一共有16×16÷8=32个字节。在字模提取的过程中,还要注意取模的顺序,顺序不同,得到的数组就不同,一般来说是从字符的左上角开始,从左向右,从上到下取模,这样程序编写上会方便一些。相同字体大小的中文字符和ASCII码字符的宽度还有所不同,一般ASCII码字符的宽度是中文字符宽度的一半,所以显示中文字符的程序和显示ASCII码字符的程序还略有不同。
 
当把一个字符取模变成一个数组后,只要对该数组中每个元素的每一位依次进行判断,对值为1的位和值为0的位进行不同的上色处理,即可完成一个字符的绘制。
 
下面的程序给出了一个简单的显示中文字符和ASCII字符的例子,我是用PCtoLCD这款软件来提取程序中想要显示的字符的。
 
#define U32 unsigned int
#define M5D(n)                          ((n) & 0x1fffff)     // To get lower 21bits
 
#define rGPCCON    (*(volatile unsigned *)0x56000020)      //Port C control
#define rGPCDAT    (*(volatile unsigned *)0x56000024)       //Port C data
#define rGPCUP     (*(volatile unsigned *)0x56000028) //Pull-up control C
 
#define rGPDCON    (*(volatile unsigned *)0x56000030)      //Port D control
#define rGPDDAT    (*(volatile unsigned *)0x56000034)       //Port D data
#define rGPDUP     (*(volatile unsigned *)0x56000038)       //Pull-up control D
 
#define rGPGCON    (*(volatile unsigned *)0x56000060)      //Port G control
#define rGPGDAT    (*(volatile unsigned *)0x56000064)       //Port G data
#define rGPGUP     (*(volatile unsigned *)0x56000068)       //Pull-up control G
 
#define rLCDCON1    (*(volatile unsigned *)0x4d000000)     //LCD control 1
#define rLCDCON2    (*(volatile unsigned *)0x4d000004)     //LCD control 2
#define rLCDCON3    (*(volatile unsigned *)0x4d000008)     //LCD control 3
#define rLCDCON4    (*(volatile unsigned *)0x4d00000c)     //LCD control 4
#define rLCDCON5    (*(volatile unsigned *)0x4d000010)     //LCD control 5
#define rLCDSADDR1  (*(volatile unsigned *)0x4d000014)    //STN/TFT Frame buffer start address 1
#define rLCDSADDR2  (*(volatile unsigned *)0x4d000018)    //STN/TFT Frame buffer start address 2
#define rLCDSADDR3  (*(volatile unsigned *)0x4d00001c)    //STN/TFT Virtual screen address set
#define rLCDINTMSK  (*(volatile unsigned *)0x4d00005c)    //LCD Interrupt mask
#define rTCONSEL     (*(volatile unsigned *)0x4d000060)   //LPC3600 Control --- edited by junon
 
#define LCD_WIDTH    320
#define LCD_HEIGHT  240
 
#define VSPW       (3-1)
#define VBPD       (15-1)
#define VFPD (12-1)
 
#define HSPW       (30-1)
#define HBPD       (38-1)
#define HFPD (20-1)
 
#define LINEVAL  (LCD_HEIGHT-1)
#define HOZVAL   (LCD_WIDTH-1)
 
//for LCDCON1
#define CLKVAL_TFT   6
#define MVAL_USED   0
#define PNRMODE_TFT      3
#define BPPMODE_TFT      13
//#define VIDEO_OUT  0
 
//for LCDCON5
#define BPP24BL   0
#define INVVCLK 0
#define INVVLINE       1
#define INVVFRAME   1
#define INVVD     0
#define INVVDEN 0
#define PWREN    1    
#define BSWP       0
#define HWSWP    0
 
volatile U32 LCD_BUFFER[LCD_HEIGHT][LCD_WIDTH];
 
unsigned char zhao[]=                  //赵
{
0x08, 0x00, 0x08, 0x00, 0x08, 0x04, 0x7E, 0x84, 0x08, 0x48, 0x08, 0x28, 0xFF, 0x10, 0x08, 0x10,
0x28, 0x28, 0x2F, 0x28, 0x28, 0x44, 0x28, 0x84, 0x58, 0x00, 0x48, 0x00, 0x87, 0xFE, 0x00, 0x00
};
unsigned char chun[]=                 //春
{
0x01, 0x00, 0x01, 0x00, 0x3F, 0xFC, 0x01, 0x00, 0x1F, 0xF8, 0x02, 0x00, 0xFF, 0xFE, 0x04, 0x20,
0x08, 0x18, 0x3F, 0xEE, 0xC8, 0x24, 0x0F, 0xE0, 0x08, 0x20, 0x08, 0x20, 0x0F, 0xE0, 0x00, 0x00
};
unsigned char jiang[]=                 //江
{
0x20, 0x00, 0x10, 0x00, 0x13, 0xFC, 0x00, 0x40, 0x88, 0x40, 0x48, 0x40, 0x50, 0x40, 0x10, 0x40,
0x10, 0x40, 0x20, 0x40, 0xE0, 0x40, 0x20, 0x40, 0x20, 0x40, 0x2F, 0xFE, 0x20, 0x00, 0x00, 0x00,
};
unsigned char ASCII_A[]=           //A
{
0x00, 0x00, 0x00, 0x10, 010, 0x18, 0x28, 0x28, 0x24, 0x3C, 0x44, 0x42, 0x42, 0xE7, 0x00, 0x00
};
unsigned char ASCII_R[]=           //R         
{
0x00, 0x00, 0x00, 0xFC, 0x42, 0x42, 0x42, 0x7C, 0x48, 0x48, 0x44, 0x44, 0x42, 0xE3, 0x00, 0x00
};
unsigned char ASCII_M[]=          //M
{
0x00, 0x00, 0x00, 0xEE, 0x6C, 0x6C, 0x6C, 0x6C, 0x54, 0x54, 0x54, 0x54, 0x54, 0xD6, 0x00, 0x00
};
 
//绘制背景
void Brush_Background( U32 c)
{
       int x,y ;
             
    for( y = 0 ; y < LCD_HEIGHT ; y++ )
    {
           for( x = 0 ; x < LCD_WIDTH ; x++ )
           {
                     LCD_BUFFER[y][x] = c ;
           }
    }
}
 
//绘制像素点
void PutPixel(U32 x,U32 y, U32 c )
{
LCD_BUFFER[y][x] = c;
}
 
//绘制大小为16×16的中文字符
void Draw_Text16(U32 x,U32 y,U32 color,const unsigned char ch[])
{
       unsigned short int i,j;
       unsigned char mask,buffer;
      
       for(i=0;i<16;i++)
       {
              mask=0x80;                  //掩码
              buffer=ch[i*2];             //提取一行的第一个字节
              for(j=0;j<8;j++)
              {                  
                     if(buffer&mask)
                     {
                            PutPixel(x+j,y+i,color);        //为笔画上色
                     }
                     mask=mask>>1;                  
              }
              mask=0x80;                  //掩码
              buffer=ch[i*2+1];         //提取一行的第二个字节
              for(j=0;j<8;j++)
              {                  
                     if(buffer&mask)
                     {
                            PutPixel(x+j+8,y+i,color);           //为笔画上色
                     }
                     mask=mask>>1;                  
              }
       }
}
 
//绘制大小为8×16的ASCII码
void Draw_ASCII(U32 x,U32 y,U32 color,const unsigned char ch[])
{
       unsigned short int i,j;
       unsigned char mask,buffer;
      
       for(i=0;i<16;i++)
       {
              mask=0x80;
              buffer=ch[i];
              for(j=0;j<8;j++)
              {                  
                     if(buffer&mask)
                     {
                            PutPixel(x+j,y+i,color);
                     }
                     mask=mask>>1;                  
              }
       }
}
 
//LCD初始化
void LCD_Init()
{
       rGPCUP  = 0x00000000;
       rGPCCON = 0xaaaa02a9;
        
       rGPDUP  = 0x00000000;
       rGPDCON=0xaaaaaaaa; //Initialize VD[15:8]
rLCDCON1=(CLKVAL_TFT<<8)|(MVAL_USED<<7)|(PNRMODE_TFT<<5)|(BPPMODE_TFT<<1)|0;
    rLCDCON2=(VBPD<<24)|(LINEVAL<<14)|(VFPD<<6)|(VSPW);
       rLCDCON3=(HBPD<<19)|(HOZVAL<<8)|(HFPD);
       rLCDCON4=(HSPW);
       rLCDCON5 = (BPP24BL<<12) | (INVVCLK<<10) | (INVVLINE<<9) | (INVVFRAME<<8) | (0<<7) | (INVVDEN<<6) | (PWREN<<3)  |(BSWP<<1) | (HWSWP);
 
       rLCDSADDR1=(((U32)LCD_BUFFER>>22)<<21)|M5D((U32)LCD_BUFFER>>1);
       rLCDSADDR2=M5D( ((U32)LCD_BUFFER+(LCD_WIDTH*LCD_HEIGHT*4))>>1 );
       rLCDSADDR3=LCD_WIDTH*4/2;
       rLCDINTMSK|=(3); // MASK LCD Sub Interrupt
       rTCONSEL = 0;     // Disable LPC3480
   
    rGPGUP=rGPGUP&(~(1<<4))|(1<<4); // Pull-up disable
       rGPGCON=rGPGCON&(~(3<<8))|(3<<8); //GPG4=LCD_PWREN
       rGPGDAT = rGPGDAT | (1<<4) ;
       rLCDCON5=rLCDCON5&(~(1<<3))|(1<<3);   // PWREN
       rLCDCON5=rLCDCON5&(~(1<<5))|(0<<5);   // INVPWREN
}
 
void Main(void)
{
      
LCD_Init();
 
rLCDCON1|=1;     //开启LCD显示
   
Brush_Background(0xFFFFFF);          //绘制白色背景
 
       //绘制黑色字符
Draw_Text16(50,100,0x0,zhao);
Draw_Text16(66,100,0x0,chun);
Draw_Text16(82,100,0x0,jiang);
Draw_ASCII(50,120,0x0,ASCII_A);
Draw_ASCII(58,120,0x0,ASCII_R);
Draw_ASCII(66,120,0x0,ASCII_M);
while(1)
{
;
}
}
 
 
       看了上面的程序,有人可能会问,如果要在程序中显示大量的中文字符,是不是要把这些字符都取模啊?回答是肯定的,但前人已经为我们完成了这一步,做成了数据库,并且进行了编码,只要按照编码规则调用该库文件,就可以检索到相要的字符。下面就来说说编码规则:每个汉字是由两个字节表示的,前一个字节表示的区号,后一个字节表示的位号,那么汉字在汉字库中的位置为:94×(区号-1)+(位号-1)。94表示的是每个区里一共有94个汉字,减1表示的是数组是从0开始,而区号和位号是从1开始的。具体到汉字在某一数据库中的位置,还需要乘以一个汉字字模所占的字节数,即[94×(区号-1)+(位号-1)]×一个汉字字模所占字节数。如一个字模大小为16×16的宋体数据库,库里每个汉字所占的字节为16×16÷8=32,则每个汉字在该宋体数据库中的位置为:[94×(区号-1)+(位号-1)]×32。ASCII码的字符调用比汉字字符要简单,只要把它乘以字模所占字节数即可找到该字符所在字库的位置,如8×16的ASCII字库,ASCII码在该字库的位置为ASCII×16。如果中文字符和ASCII码混合在一样,如何区分它们呢?其实也很简单,ASCII码的最高位是0,而中文的最高位是1,因此当读取到的一个字节的最高位是0,则该字节为ASCII码,它的下一个字节与这个字节无关;当取得到的字节的最高位是1,则表示的是中文字符,并且该字节与它的下一个字节组合在一起表示完整的一个汉字。
 
编码规则介绍完了,那么如何打开字库呢?我们可以利用前人已做好的字库,然后像访问一般文件一样打开它。另一种方法是把字库变换成一个超大的数组,那么我们就可以像操作数组一样读取字库了(在这里,我们使用的是这种方法)。
 
       下面我们就给出具体的实例,它可以显示任意的中、英文字符串。这里只给出主程序,需要调用的子程序与上面的一样。
 
#include "font_libs.h"           //内有两个数组__HZK[ ]和__ASCII[ ]
//分别表示中文字符和ASCII字符
…………
…………
void Main(void)
{
       unsigned char String[]="我的博客是:http://blog.csdn.net/zhaocj";
       int length = sizeof(String);
       int k,xx;
       unsigned char qh,wh;
       const unsigned char *mould;
LCD_Init();  
rLCDCON1|=1;
   
Brush_Background(0xffffff);
for(k=0,xx=0;k {
              if(String[k]&0x80)        //中文字符
       {
                     qh=String[k]-0xa0;               //区号
                     wh=String[k+1]-0xa0;          //位号
                     mould = & __HZK[ ( ( qh - 1 )*94 + wh- 1 )*32 ];
                     Draw_Text16(4+xx,100,0x0f,mould);
                     xx+=16;
                     k++;
       }
              else                       //ASCII码字符
              {
                     mould = & __ASCII[String[k]*16];
                     Draw_ASCII(4+xx,100,0x0,mould);
                     xx+=8;
              }
}
while(1)
{
              ;         
}
}
 
 
关键字:s3c2440  LCD  字符显示 引用地址:s3c2440的LCD字符显示

上一篇:s3c2440实时时钟的应用
下一篇:s3c2440的UART用法

推荐阅读最新更新时间:2024-03-16 14:55

LCD液晶拼接屏显示墙技术特点分析
LCD液晶大屏幕中LCD液晶拼接屏的成像方式决定了它具备:完全直角,不存在CRT显示器中所谓的几何失真的现象、不需要CRT显示器中复杂的电子枪结构,体积可以做的很小、电耗非常低、零辐射及不闪烁等优点。这些优点使得LCD液晶拼接屏在最快的时间里得到普及,而这些优点也都是所有专业领域中的用户希望利用上的,不过在某些技术指标上,LCD液晶拼接屏逊于CRT显示器的,例如响应时间、色彩丰富程度,这些缺点又使得专业用户在LCD液晶拼接屏和CRT显示器之间难以做出选择。下面上海仙视电子将LCD液晶拼接屏在几个主要的显示器性能指标上与CRT显示器进行对比,看看到底哪些差异使得专业用户在LCD显示器的选择上产生了犹豫。 液晶凭借着与生俱来的优势:
[电源管理]
Linux驱动:s3c2410/s3c2440 ts驱动分析
前言 前面结合“平台总线-设备-驱动”模型分析了ts(触摸)驱动的加载过程,现在进入驱动程序分析下其具体的实现。涉及到输入子系统、s3c2440的ADC转换和触摸控制器。 涉及的寄存器 image.png 调用probe函数 根据上一篇的分析,驱动层通过platform_driver_register注册后,会调用到该驱动层的probe函数。 s3c2410ts_probe函数分析 1. 硬件寄存器设置 获取设备参数 struct s3c2410_ts_mach_info *info; info = ( struct s3c2410_ts_mach_info *)pdev- dev.platform_data;
[单片机]
S3C2440上电启动流程概述
一、S3C2440的启动方式 1、启动介质 S3C2440在上电时会通过判断OM0和OM1的信号组合来决定指令开始执行的位置(即引导ROM的位置),同时这两个信号也用于决定BANK0(nGCS0)的总线宽度。 OM0和OM1的信号值可以由对应的两个外部引脚(引脚名称:T15和R13)的电平状态决定。 不同的启动模式 从上图可以看出,当MO =00时,S3C2440将会从挂载于Nand Flash控制器之上的外置Nand Flash启动;当 MO =01或10时,S3C2440会从挂载于内存控制器Bank0上的外部SROM上启动(指SRAM或ROM,一般Bank0上挂载的为Nor Flash,属于ROM的一种)。 2、不
[单片机]
<font color='red'>S3C2440</font>上电启动流程概述
LG Display将要淘汰LCD?重心逐渐转向OLED面板
据韩媒Business Korea报道,LG Display的LCD市场份额已跌至第四位。从LCD市场排名的下降可以看出,该公司正将重点从LCD转向OLED面板。 据悉,市场研究机构IHS Markit在日前发布的报告中称,LG Display去年第四季度在LCD面板市场的份额为14.3%,排名第四,比前一季度下降了两个等级。而LG Display在2018年第一季度之前一直是第一名,但被京东方(BOE)等中国显示器制造商挤到了第四名。 截至去年第三季,LG Display在LCD的市场份额为15.5%,仅次于京东方的18.9%。然而,在第四季度,中国台湾地区的群创电子(16.0%)和中国的CSOT电子(14.4%)领先于LG
[手机便携]
TFT LCD在GPS导航仪中的应用
概述 随着我国汽车工业的发展和汽车用户的增多,车辆的自主实时导航技术越来越受到人们的关注,并被广泛地应用GPS车载导航仪是通过接受卫星信号,再配合电子地图数据,来适时掌握自己的方位与目的地的一种嵌入式产品,自主导航的模式不收取任何使用费,用户可以根据自己的需要有选择地获得地图数据。 显示系统模块是GPS导航仪的重要模块,其中的TFT液晶显示器色彩更逼真,更平滑细腻,层次感更强,而且具有体积小、重量轻等优点,本设计采用samsung公司的LTV350QV-F05 TFT LCD屏。其系统框图如图1所示。 XSCAL PXA255处理器的LCD控制接口 PXA255处理器是Intel公司推出的一款基于XSC
[汽车电子]
全球大尺寸TFT LCD面板营业额在2014年二月触底
NPD DisplaySearch上海办公室,2014年3月31日---NPD DisplaySearch通过统计每个月的全球面板营收看出,大尺寸薄膜晶体管液晶显示(TFT LCD)行业最近几年正处于最具挑战的时期。根据NPD DisplaySearch最新出版的TFT LCD月度出货量报告Monthly TFT LCD Shipment Database显示,2014年大尺寸(9寸及以上)TFT LCD的营收在二月跌到了三年来的最低点,也是2011年以来第一次单月低于50亿美元,同比下降3%M/M,13%Y/Y。 图一, 全球大尺寸(9寸及以上)TFT LCD的月营收金额 来源:NPD DisplaySearch
[家用电子]
全球大尺寸TFT <font color='red'>LCD</font>面板营业额在2014年二月触底
可弯曲LCD降临 柔软不再是OLED专利
      OLED具有可弯曲的特点,很多人认为OLED电视将是电视产业的未来之星而不是液晶电视或等离子电视。但近日由13家公司和机构组成的日本财团正在努力开发超薄、可弯曲的液晶显示器(LCD)。该公司表示他们已经找到了一种新的方法制作液晶显示器,他们使用塑料薄膜代替玻璃基板,这将使液晶显示器将更柔软、更薄。       日本先进显示材料研究协会是财团的成员之一(包括日立和夏普)。理论上说,利用多层塑料薄膜显示颜色可以代替玻璃基板,但这是一个复杂的过程。       协会表示,其新的高密度生产工艺结合了其它类型的薄膜彩色滤光片。显然,该生产方法需要更少的过程,这意味着生产商需要更少的设备。协会还表示新的工艺可以减少75%的
[家用电子]
AVR LCD12864 程序
/********************************************************************* 目 的: 建立LCD12864操作库 目标系统: 基于AVR单片机 应用软件: ICCAVR 版 本: Version 1.0 *********************************************************************/ /*010101010101010101
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved