找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 10009|回复: 7
打印 上一主题 下一主题
收起左侧

合泰单片机HT66F70A LCD12864液晶屏驱动 汇编和c语言都有

[复制链接]
跳转到指定楼层
楼主
12864液晶显示屏单片机驱动,单片机型号是合泰HT66F70A


完整代码下载: 12864驱动1.zip (41.58 KB, 下载次数: 114)


c语言版本的程序:
  1. #include <HT66F70A.H>
  2. #include "delay.h"

  3. #define IO_12864   _pc //定义12864的数据端口
  4. #define IO_12864_C   _pcc

  5. #define RS_12864  _pd5 //定义命令数据寄存器选择端
  6. #define RW_12864  _pd6 //定义定义读写控制端        
  7. #define E_12864   _pd7  //定义使能端
  8. #define RST_12864  _pd4 //定义复位端
  9. //#define CS2   _pd3
  10. //#define CS1   _pd2



  11. void delay_12864(unsigned int ); //延时子函数原型
  12. /*void busychk_12864(void); //忙检测子函数原型*/
  13. void wrtcom_12864(unsigned char); //写命令子函数原型
  14. unsigned char reddat_12864(void); //读数据子函数原型
  15. void wrtdat_12864(unsigned char); //写数据子函数原型
  16. //void choose_12864(unsigned char i);        //选屏
  17. //void drawpic_12864( unsigned char*); //绘图子函数原型
  18. //void drawdot_12864(unsigned char,unsigned char,unsigned char);//画点子函数原型
  19. //void drawrec_12864(unsigned char,unsigned char,unsigned char,unsigned char,unsigned char);  //画矩形子函数原型
  20. void clnGDR_12864(void); //清空GDRAM子函数原型
  21. void inputchars_12864(unsigned char,unsigned char,unsigned char*);//输入短字符串子函数原型
  22. void scroll_12864(unsigned char*); //输入长字符串卷动子函数原型
  23. void initial_12864(void); //12864初始化子函数原型


  24. void main()
  25. {  
  26.         unsigned char  uctech[] = {"wweerrtt"};
  27.         _pcc=0;
  28.         _pdc=0;
  29.         //_pcpu=0xff;
  30.         _pc=0;
  31.         _pd=0;

  32.         initial_12864();
  33.         clnGDR_12864();
  34.    inputchars_12864(1,1,uctech);
  35.         

  36. }
  37.                                                                 /***********        延时子程序,最大延迟时间为65ms***************/
  38.                                                                                 
  39. void delay_12864(unsigned int del){
  40. unsigned int i;
  41. for(i = 0; i < del; i++){;}
  42. }


  43.                                                                         /***************忙检测,若忙则等待,最长等待时间为60ms**********/
  44.                                                                                 
  45. //                                                                                
  46. //void busychk_12864(void){
  47. //unsigned int timeout = 0;
  48. //IO_12864_C=1;/////////
  49. //E_12864 = 0;
  50. //RS_12864 = 0;
  51. //RW_12864 = 1;
  52. //E_12864 = 1;
  53. //while((IO_12864 & 0x80) && ++timeout != 0); //忙状态检测,等待超时时间为60ms
  54. //E_12864 = 0;
  55. //}


  56.                                                                                                 /****************写命令子程序*********************/
  57.                                                                                                 
  58.                                                                                                 
  59.                                                                                                 
  60. void wrtcom_12864(unsigned char com){
  61. //busychk_12864();
  62. IO_12864_C=0; ///////////////////////////
  63. delay_12864(5000);////////
  64. E_12864 = 0;
  65. RS_12864 = 0;
  66. RW_12864 = 0;
  67. E_12864 = 1;
  68. IO_12864 = com;
  69. delay_12864(5000); //使能延时50us!!!注意这里,如果是较快的CPU应该延时久一些
  70. E_12864 = 0;
  71. }

  72.                                                                                         /**************************读数据子程序,返回一个字节的数据****************/
  73.                                                                                        
  74.                                                                                        
  75. unsigned char reddat_12864(void){
  76. unsigned char temp;
  77. //busychk_12864();
  78. delay_12864(5000);/////////////
  79. E_12864 = 0;
  80. IO_12864_C=0;/////////////////////
  81. IO_12864 = 0xff; //IO口置高电平,读引脚
  82. RS_12864 = 1;
  83. RW_12864 = 1;
  84. E_12864 = 1;
  85. delay_12864(5000); //使能延时50us!!!注意这里,如果是较快的CPU应该延时久一些
  86. IO_12864_C=1;////////////////////////
  87. temp = IO_12864;
  88. return temp;
  89. }



  90.                                                                                                         /*****************写数据子程序***********************/
  91.                                                                                                         
  92.                                                                                                         
  93.                                                                                                         
  94. void wrtdat_12864(unsigned char dat){
  95. //busychk_12864();
  96. IO_12864_C=0;//////////////////
  97. delay_12864(5000);/////////////
  98. E_12864 = 0;
  99. RS_12864 = 1;
  100. RW_12864 = 0;
  101. E_12864 = 1;
  102. IO_12864 = dat;
  103. delay_12864(5000); //使能延时50us!!!注意这里,如果是较快的CPU应该延时久一些
  104. E_12864 = 0;
  105. }


  106.                                                                     /***************选屏****************/
  107.                                                                     
  108.                                                                     
  109. //void choose_12864(unsigned char i)//i是要写的屏.0是左屏,1是右屏,2是双屏;
  110. //{                                                                         //此处在硬件上运行时i的电平全部与程序相反;
  111. //  switch (i)                                                 //此版本为仿真版;
  112. //  {
  113. //    case 0: CS1=0;CS2=1;break;        //比如此处如果要在电路上运行则应该改为CS=1;CS2=0;   
  114. //    case 1: CS1=1;CS2=0;break;
  115. //        case 2: CS1=0;CS2=0;break;
  116. //        default: break;
  117. //  }
  118. //}


  119.                                                 /************显示图片函数,参数时图片的地址,图片必须是128*64像素的逐行逐行的点阵信息。****************/
  120.                                                 
  121.                                                 
  122.                                                 
  123. //void drawpic_12864( unsigned char   *pPicture ){
  124. //unsigned char j, k ;
  125. //wrtcom_12864(0x34);   //在写GDRAM的地址之前一定要打开扩充指令集,否则地址写不进去!!
  126. //for( j = 0 ; j < 32 ; j++ )
  127. //{
  128. //wrtcom_12864(0x80 + j) ; //写Y 坐标
  129. //wrtcom_12864(0x80) ; //写X 坐标
  130. //for( k = 0 ; k < 16 ; k++ )         //写一整行数据
  131. //{
  132. //wrtdat_12864( *pPicture++ );
  133. //}
  134. //}
  135. //for( j = 0 ; j < 32 ; j++ )
  136. //{
  137. //wrtcom_12864(0x80 + j) ; //写Y 坐标
  138. //wrtcom_12864(0x88) ; //写X 坐标
  139. //for( k = 0 ; k < 16 ; k++ )         //写一整行数据
  140. //{
  141. //wrtdat_12864( *pPicture++ ) ;
  142. //}
  143. //}
  144. //wrtcom_12864(0x36); //打开绘图显示
  145. //wrtcom_12864(0x30); //返回基本指令集
  146. //}
  147. //   
  148. ////画点函数,左上角为参考点(0,0)
  149. ////右下角为(127,63),点坐标形式为(行坐标,列坐标)
  150. ////参数type用于设置画黑点、白点或取反(黑变白,白变黑)
  151. ////type = 0为白色,1 为黑色,2为取反
  152. ////画完点不开启显示,需要自行开启绘图显示
  153. //void drawdot_12864(unsigned char y,unsigned char x,unsigned char type){
  154. //unsigned char X,Y,k; //X存储行地址,Y存储列地址,k存储点在字中的位置(0~15从左至右)
  155. //unsigned char DH,DL;
  156. //if(y >= 0 && y <= 63 && x >= 0 && x <= 127) {
  157. //if(y < 32){ //确定所画点的地址行与列地址
  158. //X = 0x80 + (x >> 4);
  159. //Y = 0x80 + y;
  160. //}else{
  161. //X = 0x88 + (x >> 4);
  162. //Y = 0x80 + (y - 32);
  163. //}
  164. //wrtcom_12864(0x34); //开启扩展指令,关闭绘图显示
  165. //wrtcom_12864(Y); //写行位地址
  166. //wrtcom_12864(X); //写列字地址
  167. //DH = reddat_12864(); //假读
  168. //DH = reddat_12864();   //读高字节
  169. //DL = reddat_12864(); //读低字节
  170. //k = x % 16;
  171. //switch(type){ //判断类型,是黑点还是白点还是取反
  172. //case 0: //画白点
  173. //  if(k < 8){
  174. //DH &= ~(0x01 << (7 - k));
  175. //}else{
  176. //DL &= ~(0x01 << (7 - (k % 8)));
  177. //}
  178. //break;
  179. //case 1: //画黑点
  180. //if(k < 8){
  181. //DH |= (0x01 << (7 - k));
  182. //}else{
  183. //DL |= (0x01 << (7 - (k % 8)));
  184. //}
  185. //break;
  186. //case 2: //对点取反
  187. //if(k < 8){
  188. //DH ^= (0x01 << (7 - k));
  189. //}else{
  190. //DL ^= (0x01 << (7 - (k % 8)));
  191. //}
  192. //break;
  193. //default:
  194. //break;  
  195. //}
  196. //wrtcom_12864(Y); //写行位地址
  197. //wrtcom_12864(X);   //写列字地址
  198. //wrtdat_12864(DH); //回写高字节
  199. //wrtdat_12864(DL); //回写低字节
  200. //
  201. //wrtcom_12864(0x36); //开启绘图显示
  202. //wrtcom_12864(0x30); //转回普通指令
  203. //}
  204. //}
  205. //
  206. //                                                        /**********************画矩形子函数,参数为(点1行坐标,点1列坐标,点2行坐标,点2列坐标,*********
  207. //                                                        **********************线条颜色(0为白,1为黑,2对原色取反))*************************************/  
  208. //                        
  209. //                        
  210. //                        
  211. //void drawrec_12864(unsigned char y1,unsigned char x1,unsigned char y2,unsigned char x2,unsigned char type){
  212. //unsigned char largex,largey,smallx,smally;
  213. //unsigned char i;
  214. //if(x1 > x2){
  215. //largex = x1;
  216. //smallx = x2;
  217. //}else{
  218. //largex = x2;
  219. //smallx = x1;
  220. //}
  221. //if(y1 > y2){
  222. //largey = y1;
  223. //smally = y2;
  224. //}else{
  225. //largey = y2;
  226. //smally = y1;
  227. //}
  228. ////画4条矩形边框
  229. //for(i = smallx; i < largex; i++){
  230. //drawdot_12864(largey,i,type);
  231. //}
  232. //for(i = largey; i > smally; i--){
  233. //drawdot_12864(i,largex,type);
  234. //}
  235. //for(i = largex; i > smallx; i--){
  236. //drawdot_12864(smally,i,type);
  237. //}
  238. //for(i = smally; i < largey; i++){
  239. //drawdot_12864(i,smallx,type);
  240. //}
  241. //wrtcom_12864(0x30); //返回普通指令
  242. //}

  243.                                                                                                         /*****************清空GDRAM,往GDRAM内部写满0x00*************/
  244.                                                                                                         
  245.                                                                                                         
  246. void clnGDR_12864(void){
  247. unsigned char j,k;
  248.   wrtcom_12864(0x34);//在写GDRAM的地址之前一定要打开扩充指令集,否则地址写不进去!!
  249. for( j = 0 ; j < 32 ; j++ )
  250. {
  251. wrtcom_12864(0x80 + j) ; //写Y 坐标
  252. wrtcom_12864(0x80) ; //写X 坐标
  253. for( k = 0 ; k < 32 ; k++ ) //写一整行数据
  254. {
  255. wrtdat_12864( 0x00 );
  256. }
  257. }
  258. wrtcom_12864(0x30); //返回基本指令集
  259. }



  260.                                                         /***********************写入短字符串函数,参数分别为显示行(0~3),显示列(0~7),字符串首地址
  261.                                                                                                         字符串只会在一行显示,超过的不显示**********************************/
  262.                                                                                                         
  263.                                                                                                         

  264. void inputchars_12864(unsigned char y,unsigned char x,unsigned char *dataaddr){
  265. unsigned char i,address=0;
  266. switch (y){ //设置字符显示起始地址
  267. case 0: address = 0x80 + x;
  268. break;
  269. case 1: address = 0x90 + x;
  270. break;
  271. case 2: address = 0x88 + x;
  272. break;
  273. case 3: address = 0x98 + x;
  274. break;
  275. default: break;
  276. }
  277. wrtcom_12864(0x30);
  278. wrtcom_12864(address);
  279. for(i = x;i < 8; i++){
  280. if(*dataaddr != '\0'){
  281. wrtdat_12864(*dataaddr++);
  282. }
  283. }
  284. }

  285.                                                                                         /******************长字符串显示卷屏函数**************************/
  286.                                                                                        
  287.                                                                                        
  288.                                                                                        
  289. void scroll_12864(unsigned char *ser){
  290.         //addr存储地址的中间变量,flag卷屏地址,hang要写数据的行,
  291.         //over写完字符串后继续写的空字符计数
  292. unsigned char i,addr,flag,hang,over,*ptdat;
  293. ptdat = ser; //获得字符串首地址
  294. over = 0; //写入空字符串(写完字符串后)数目初始化为0
  295. wrtcom_12864(0x80); //写第一行字符
  296. for(i = 0; i < 16; i++){
  297. if(*ptdat != '\0'){
  298. wrtdat_12864(*ptdat++);
  299. }else{
  300. wrtdat_12864(0x20);
  301. over++;
  302. }
  303. }
  304. wrtcom_12864(0x90); //写第二行字符
  305. for(i = 0; i < 16; i++){
  306. if(*ptdat != '\0'){
  307. wrtdat_12864(*ptdat++);
  308. }else{
  309. wrtdat_12864(0x20);
  310. over++;
  311. }
  312. }
  313. wrtcom_12864(0x88); //写第三行字符
  314. for(i = 0; i < 16; i++){
  315. if(*ptdat != '\0'){
  316. wrtdat_12864(*ptdat++);
  317. }else{
  318. wrtdat_12864(0x20);
  319. over++;
  320. }
  321. }
  322. wrtcom_12864(0x98); //写第四行字符
  323. for(i = 0; i < 16; i++){
  324. if(*ptdat != '\0'){
  325. wrtdat_12864(*ptdat++);
  326. }else{
  327. wrtdat_12864(0x20);
  328. over++;
  329. }
  330. }
  331. ptdat = ptdat - 32;
  332. wrtcom_12864(0xa0); //写第三行DDRAM,写入内容为第三行字符串和第五行字符串
  333. for(i = 0; i < 16; i++){ //写第三行字符串
  334. if(*ptdat != '\0'){
  335. wrtdat_12864(*ptdat++);
  336. }else{
  337. wrtdat_12864(0x20);
  338. over++;
  339. }
  340. }
  341. ptdat = ptdat + 16;
  342. for(i = 0; i < 16; i++){   //写第五行字符串
  343. if(*ptdat != '\0'){
  344. wrtdat_12864(*ptdat++);
  345. }else{
  346. wrtdat_12864(0x20);
  347. over++;
  348. }
  349. }
  350. wrtcom_12864(0x0c); //开显示
  351. if(over > 15){;} //显示字符不足4行,不卷动
  352. else //显示字符大于4行,继续写字符,同时开启卷动
  353. {
  354. hang = 4;
  355. flag = 0x01;
  356. while(1){
  357. switch(hang){   //设置写入DDRAM的地址
  358. case 1: addr = 0x80; break;
  359. case 2: addr = 0x90; break;
  360. case 3: addr = 0xa0; break;
  361. case 4: addr = 0xb0; break;
  362. }
  363. switch(hang){ //设置要写数据的下一行
  364. case 1: hang = 2; break;
  365. case 2: hang = 3; break;
  366. case 3: hang = 4; break;
  367. case 4: hang = 1; break;
  368. }
  369. ptdat = ptdat - 32;
  370. for(i = 0; i < 8; i++){
  371. wrtcom_12864(0x34); //打开扩展指令
  372. wrtcom_12864(0x03); //允许输入卷动地址
  373. wrtcom_12864(0x40 + flag++); //设置卷动地址
  374. wrtcom_12864(0x30);         //回到基本指令
  375. wrtcom_12864(addr + i);
  376. delay_12864(20000);
  377. if(*ptdat != '\0'){
  378. wrtdat_12864(*ptdat++);   //连续写入两个字节之高字节
  379. }else{
  380. wrtdat_12864(0x20);
  381. }
  382. if(*ptdat != '\0'){
  383. wrtdat_12864(*ptdat++);   //连续写入两个字节之低字节
  384. }else{
  385. wrtdat_12864(0x20);
  386. }
  387. }
  388. ptdat = ptdat + 16;
  389. for(i = 8; i < 16; i++){
  390. wrtcom_12864(0x34); //打开扩展指令
  391. wrtcom_12864(0x03); //允许输入卷动地址
  392. if(flag == 64){flag = 0;}
  393. wrtcom_12864(0x40 + flag); //设置卷动地址
  394. flag++;
  395. wrtcom_12864(0x30); //回到基本指令
  396. wrtcom_12864(addr + i);
  397. delay_12864(20000);
  398. if(*ptdat != '\0'){
  399. wrtdat_12864(*ptdat++);  //连续写入两个字节之高字节
  400. }else{
  401. over++;  //写完最后一行字符,需要再卷动16次才能显示出来。
  402. wrtdat_12864(0x20);
  403. }
  404. if(*ptdat != '\0'){
  405. wrtdat_12864(*ptdat++);    //连续写入两个字节之低字节
  406. }else{
  407. wrtdat_12864(0x20);
  408. }
  409. }
  410. if(over < 8){;}   //最后一行显示完毕则停止卷动
  411. else {break;}
  412. }
  413. }
  414. }

  415.                                                                         /***********************初始化12864子函数***********************************/
  416.                                                                         
  417.                                                                         
  418.                                                                         
  419.                                                                         
  420.                                                                         
  421. void initial_12864(void){
  422. delay_12864(40000);
  423. RST_12864 = 1;
  424. RST_12864 = 0; //复位
  425. delay_12864(500);
  426. RST_12864 = 1;

  427. wrtcom_12864(0x30); //设置为基本指令集动作
  428. delay_12864(100);
  429. wrtcom_12864(0x30); //设置为基本指令集动作
  430. delay_12864(37);
  431. wrtcom_12864(0x08); //设置显示、光标、闪烁全关。
  432. delay_12864(100);
  433. wrtcom_12864(0x01); //清屏,并且DDRAM数据指针清零
  434. delay_12864(1000);
  435. wrtcom_12864(0x06);   //进入模式设置
  436. wrtcom_12864(0x0c);     //开显示
  437. }










复制代码



汇编语言版本的程序:

  1. ; Generated by holtek-gcc v3.13, Wed Apr 13 19:10:57 2016
  2. ;
  3. ; Configuration:
  4. ;       with long instruction
  5. ;       Multi-ROM, Multi-RAM
  6. ;       bits_per_rom_unit:16
  7. ;       with mp1
  8. ;       with tbhp, address(0x9)
  9. ;          Use tabrd-const
  10. ;      
  11. ; SFR address of long-instruction arch:
  12. ;    mp0 = -1,1,0
  13. ;    mp1 = 4,3,2
  14. ;    mp2 = 14,13,12
  15. ;    tbp = 9,7,8
  16. ;    acc = 5
  17. ;    pcl = 6
  18. ;    status = 10
  19. ;    bp = 11
  20. ;    intc = 48
  21. ;      
  22. ; use 'tabrdc' instead of 'tabrd'
  23. ;      

  24. #pragma translator "holtek-gcc 4.6.4" "3.13" "build 20130711"
  25. ; Rebuild 26

  26. ir equ [2]
  27. mp equ [3]
  28. sbp equ [4]
  29. acc equ [5]
  30. bp equ [11]
  31. tblp equ [7]
  32. tbhp equ [9]
  33. status equ [10]
  34. c equ [10].0
  35. ac equ [10].1
  36. z equ [10].2
  37. ov equ [10].3
  38. cz equ [10].6
  39. sc equ [10].7
  40. intc equ [48]

  41. extern ra:byte
  42. extern rb:byte
  43. extern rc:byte
  44. extern rd:byte
  45. extern re:byte
  46. extern rf:byte
  47. extern rg:byte
  48. extern rh:byte
  49. extern _Crom2Prom:near
  50. extern _Crom2PromNext:near
  51. @HCCINIT        .section 'data'
  52. @HCCINIT0        .section 'data'

  53. #pragma debug scope 1 1
复制代码


评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享淘帖 顶1 踩
回复

使用道具 举报

沙发
ID:116003 发表于 2016-4-22 20:30 | 只看该作者
谢谢                                 
回复

使用道具 举报

板凳
ID:124260 发表于 2016-5-31 15:56 | 只看该作者
本帖最后由 帅帅 于 2016-5-31 15:57 编辑

大神,我想用合泰HT66F70A来控制4X4然后显示在lcd1602程序怎么写啊?
回复

使用道具 举报

地板
ID:166387 发表于 2017-2-24 13:48 | 只看该作者
下载看看有没有学习价值
回复

使用道具 举报

5#
ID:186647 发表于 2017-4-6 13:02 | 只看该作者
正好手头用的是70a的单片机,
回复

使用道具 举报

6#
ID:190835 发表于 2017-5-5 10:46 | 只看该作者
竟然少
回复

使用道具 举报

7#
ID:260698 发表于 2018-3-30 16:57 | 只看该作者
12864怎么接合泰66f70a,有电路图没?
回复

使用道具 举报

8#
ID:296136 发表于 2018-5-14 16:45 | 只看该作者
楼主,你有HT66F70A LCD12864显示时间或者温度的程序吗
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表