PIC18F4620 2004屏显示乱码

[复制链接]
1744|1
 楼主| yaooxp 发表于 2013-1-30 19:21 | 显示全部楼层 |阅读模式


PIC18F4620单片机,控制2004字符屏,有的时候屏幕显示正常,有的时候出现乱码,百度了一下,说是初始化程序有问题,但我对PIC的单片机不了解,请高手帮我修改一下初始化程序,谢谢。下面是液晶屏商家提供的C语言例。
  1. 标准型16X2液晶显示字符模块
  2. 1602采用标准的16脚接口,其中:第1脚:VSS为地电源第2脚:VDD接5V正电源第3脚:V0为液晶显示器对比度调整端,接正电源时对比度最弱,接地电源时对比度最高,对比度过高时会产生“鬼影”,使用时可以通过一个10K的电位器调整对比度第4脚:RS为寄存器选择,高电平时选择数据寄存器、低电平时选择指令寄存器。第5脚:R/W为读写信号线,高电平时进行读操作,低电平时进行写操作。当RS和RW共同为低电平时可以写入指令或者显示地址,当RS为低电平RW为高电平时可以读忙信号,当RS为高电平RW为低电平时可以写入数据。第6脚:E端为使能端,当E端?筛叩缙教涑傻偷缙绞保壕?橹葱忻睢?第7~14脚:D0~D7为8位双向数据线。第15脚:背光电源正极第16脚:背光电源负极1602液晶模块内部的字符发生存储器(CGROM)已经存储了160个不同的点阵字符图形,如表1所示,这些字符有:阿拉伯数字、英文字母的大小写、常用的符号、和日文假名等,每一个字符都有一个固定的代码,比如大写的英文字母“A”的代码是01000001B(41H),显示时模块把地址41H中的点阵字符图形显示出来,我们就能看到字母“A” 。以下是在液晶模块的第二行第一个字符的位置显示字母“A”的程序: ORG 0000HRS EQU P3.7;确定具体硬件的连接方式
  3. RW EQU P3.6 ;确定具体硬件的连接方式
  4. E EQU P3.5 ;确定具体硬件的连接方式MOV P1,#00000001B ;清屏并光标复位
  5. ACALL ENABLE;调用写入命令子程序MOV P1,#00111000B ;设置显示模式:8位2行5x7点阵
  6. ACALL ENABLE ;调用写入命令子程序MOV P1,#00001111B ;显示器开、光标开、光标允许闪烁
  7. ACALL ENABLE ;调用写入命令子程序MOV P1,#00000110B ;文字不动,光标自动右移
  8. ACALL ENABLE ;调用写入命令子程序MOV P1,#0C0H ;写入显示起始地址(第二行第一个位置)
  9. ACALL ENABLE ;调用写入命令子程序MOV P1,#01000001B ;字母A的代码SETB RS ;RS=1
  10. CLR RW ;RW=0 ;准备写入数据
  11. CLR E ;E=0 ;执行显示命令
  12. ACALL DELAY ;判断液晶模块是否忙?
  13. SETB E ;E=1 ;显示完成,程序停车AJMP $ENABLE:
  14. CLR RS ;写入控制命令的子程序
  15. CLR RW
  16. CLR E
  17. ACALL DELAY
  18. SETB E
  19. RETDELAY:
  20. MOV P1,#0FFH ;判断液晶显示器是否忙的子程序
  21. CLR RS
  22. SETB RW
  23. CLR E
  24. NOP
  25. SETB E
  26. JB P1.7,DELAY ;如果P1.7为高电平表示忙就循环等待
  27. RETEND    程序在开始时对液晶模块功能进行了初始化设置,约定了显示格式。注意显示字符时光标是自动右移的,无需人工干预,每次输入指令都先调用判断液晶模块是否忙的子程序DELAY,然后输入显示位置的地址0C0H,最后输入要显示的字符A的代码41H。
  28. SMC1602A(16*2)模拟口线接线方式
  29. 连接线图:
  30.        ---------------------------------------------------
  31.        |LCM-----51   | LCM-----51   | LCM------51      |
  32.        ------------------------------------------------|
  33.        |DB0-----P1.0 | DB4-----P1.4 | RW-------P2.0    |
  34.        |DB1-----P1.1 | DB5-----P1.5 | RS-------P2.1    |
  35.        |DB2-----P1.2 | DB6-----P1.6 | E--------P2.2    |
  36.        |DB3-----P1.3 | DB7-----P1.7 | VLCD接1K电阻到GND|
  37.        ---------------------------------------------------
  38. [注:AT89S52使用12M晶振]
  39. =============================================================*/#define LCM_RW  P2_0 //定义引脚
  40. #define LCM_RS  P2_1
  41. #define LCM_E   P2_2
  42. #define LCM_Data  P1
  43. #define Busy    0x80 //用于检测LCM状态字中的Busy标识#i ncludevoid WriteDataLCM(unsigned char WDLCM);
  44. void WriteCommandLCM(unsigned char WCLCM,BuysC);
  45. unsigned char ReadDataLCM(void);
  46. unsigned char ReadStatusLCM(void);
  47. void LCMInit(void);
  48. void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData);
  49. void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData);
  50. void Delay5Ms(void);
  51. void Delay400Ms(void);unsigned char code uctech[] = {"uctech"};
  52. unsigned char code net[] = {"uctech.icpcn.com"};void main(void)
  53. {
  54. Delay400Ms(); //启动等待,等LCM讲入工作状态
  55. LCMInit(); //LCM初始化
  56. Delay5Ms(); //延时片刻(可不要) DisplayListChar(0, 5, uctech);
  57. DisplayListChar(0, 0, net);
  58. ReadDataLCM();//测试用句无意义
  59. while(1);
  60. }//写数据
  61. void WriteDataLCM(unsigned char WDLCM)
  62. {
  63. ReadStatusLCM(); //检测忙
  64. LCM_Data = WDLCM;
  65. LCM_RS = 1;
  66. LCM_RW = 0;
  67. LCM_E = 0; //若晶振速度太高可以在这后加小的延时
  68. LCM_E = 0; //延时
  69. LCM_E = 1;
  70. }//写指令
  71. void WriteCommandLCM(unsigned char WCLCM,BuysC) //BuysC为0时忽略忙检测
  72. {
  73. if (BuysC) ReadStatusLCM(); //根据需要检测忙
  74. LCM_Data = WCLCM;
  75. LCM_RS = 0;
  76. LCM_RW = 0;
  77. LCM_E = 0;
  78. LCM_E = 0;
  79. LCM_E = 1;
  80. }//读数据
  81. unsigned char ReadDataLCM(void)
  82. {
  83. LCM_RS = 1;
  84. LCM_RW = 1;
  85. LCM_E = 0;
  86. LCM_E = 0;
  87. LCM_E = 1;
  88. return(LCM_Data);
  89. }//读状态
  90. unsigned char ReadStatusLCM(void)
  91. {
  92. LCM_Data = 0xFF;
  93. LCM_RS = 0;
  94. LCM_RW = 1;
  95. LCM_E = 0;
  96. LCM_E = 0;
  97. LCM_E = 1;
  98. while (LCM_Data & Busy); //检测忙信号
  99. return(LCM_Data);
  100. }void LCMInit(void) //LCM初始化
  101. {
  102. LCM_Data = 0;
  103. WriteCommandLCM(0x38,0); //三次显示模式设置,不检测忙信号
  104. Delay5Ms();
  105. WriteCommandLCM(0x38,0);
  106. Delay5Ms();
  107. WriteCommandLCM(0x38,0);
  108. Delay5Ms();  WriteCommandLCM(0x38,1); //显示模式设置,开始要求每次检测忙信号
  109. WriteCommandLCM(0x08,1); //关闭显示
  110. WriteCommandLCM(0x01,1); //显示清屏
  111. WriteCommandLCM(0x06,1); // 显示光标移动设置
  112. WriteCommandLCM(0x0C,1); // 显示开及光标设置
  113. }//按指定位置显示一个字符
  114. void DisplayOneChar(unsigned char X, unsigned char Y, unsigned char DData)
  115. {
  116. Y &= 0x1;
  117. X &= 0xF; //限制X不能大于15,Y不能大于1
  118. if (Y) X |= 0x40; //当要显示第二行时地址码+0x40;
  119. X |= 0x80; // 算出指令码
  120. WriteCommandLCM(X, 0); //这里不检测忙信号,发送地址码
  121. WriteDataLCM(DData);
  122. }//按指定位置显示一串字符
  123. void DisplayListChar(unsigned char X, unsigned char Y, unsigned char code *DData)
  124. {
  125. unsigned char ListLength;  ListLength = 0;
  126. Y &= 0x1;
  127. X &= 0xF; //限制X不能大于15,Y不能大于1
  128. while (DData[ListLength]>0x20) //若到达字串尾则退出
  129.   {
  130.    if (X <= 0xF) //X坐标应小于0xF
  131.     {
  132.      DisplayOneChar(X, Y, DData[ListLength]); //显示单个字符
  133.      ListLength++;
  134.      X++;
  135.     }
  136.   }
  137. }//5ms延时
  138. void Delay5Ms(void)
  139. {
  140. unsigned int TempCyc = 5552;
  141. while(TempCyc--);
  142. }//400ms延时
  143. void Delay400Ms(void)
  144. {
  145. unsigned char TempCycA = 5;
  146. unsigned int TempCycB;
  147. while(TempCycA--)
  148. {
  149.   TempCycB=7269;
  150.   while(TempCycB--);
  151. };
  152. 以上程序仅供参考!!
  153.   

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?注册

×
yewuyi 发表于 2013-1-31 11:30 | 显示全部楼层
自己的程序只能自己检查,但从描述和图纸看,基本肯定是代码问题。

检查自己的LCD显示数据发送时序是否被中断等干扰了?
您需要登录后才可以回帖 登录 | 注册

本版积分规则

1

主题

1

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部