找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 6483|回复: 7
收起左侧

stm32串行st7920 128*64屏显示字符源程序

[复制链接]
ID:220633 发表于 2017-7-18 20:10 | 显示全部楼层 |阅读模式
stm32单片机串行方式控制st7920 128*64屏显示字符,附件里面有2个程序,全部都是经过测试ok的
b00d5298a8551fe631848e1e1e5d996a_604.jpg

单片机源程序如下:
  1. /**********************************************************
  2. * @ File name -> lcd12864.c
  3. * @ Version   -> V1.1.2
  4. * @ Date      -> 11-30-13
  5. * @ Brief     -> LCD12864驱动函数
  6. *                适用于ST7920驱动芯片

  7. V1.1
  8. * @ Revise    -> A、修正操作命令宏定义。
  9. * @           -> B、增加串口、并口转换宏定义,增加对高速度MCU控制通讯频率延时函数
  10. * @           -> C、显示字符串函数增加显示长度选择,即显示多少个ASCII可显示字符,汉字x2即可
  11. * @           -> D、去掉显示数组函数,合并在字符串显示里

  12. V1.1.1
  13. * @ Revise    -> 增加字符串显示函数,可以从要显示的数组中任意位置显示一定长度字符串

  14. V1.1.2
  15. * @ Revise    -> A、增加对低速晶振系统的MCU通讯频率是否使用的宏定义
  16. * @           -> B、增加在检测液晶忙超时退出,预防死循环
  17. **********************************************************/

  18. #include "lcd12864.h"

  19. /**********************************************************
  20.                     自定义显示字符                        
  21. **********************************************************/

  22. #define CGRAM_Value   64        //写入CGRAM数量
  23.   
  24. u8 code CGROM_Code[] = {
  25.                                                 //天线图形       
  26.                                                 0xff,0xff,0x80,0x01,0xbf,0xfd,0xdf,0xfb,0xec,0x37,0xf6,0x6f,0xfb,0xdf,0xfd,0xbf,
  27.                                                 0xfe,0x7f,0xfe,0x7f,0xfe,0x7f,0xfe,0x7f,0xfe,0x7f,0xfe,0x7f,0xfe,0x7f,0xff,0xff,

  28.                                                 //摄氏图形
  29. //                                                0x00,0x00,0x00,0x00,0x00,0x00,0x1c,0x70,0x14,0xd8,0x1d,0x88,0x01,0x80,0x01,0x80,
  30. //                                                0x01,0x80,0x01,0x80,0x01,0x88,0x00,0xc8,0x00,0x70,0x00,0x00,0x00,0x00,0x00,0x00,   

  31.                                                 //蓝牙图形
  32.                                                 0x07,0xe0,0x0e,0x70,0x1e,0xb8,0x3e,0xdc,0x36,0xec,0x3a,0xdc,0x3c,0xbc,0x3e,0x7c,
  33.                                                 0x3e,0x7c,0x3c,0xbc,0x3a,0xdc,0x36,0xec,0x3e,0xdc,0x1e,0xb8,0x0e,0x70,0x07,0xe0,

  34.                                                 //CD图形
  35.                                                 0xff,0xff,0xff,0xff,0xff,0xff,0xe3,0x07,0xc9,0xb3,0x9d,0xb9,0x9f,0xb9,0x9f,0xb9,
  36.                                                 0x9f,0xb9,0x9f,0xb9,0x9d,0xb9,0xc9,0xb3,0xe3,0x07,0xff,0xff,0xff,0xff,0xff,0xff,

  37.                                                 //右指向手型
  38.                                                 0x00,0x00,0x1e,0x00,0x12,0x00,0x73,0xfc,0x52,0x02,0x52,0x1c,0x53,0xe0,0x52,0x10,
  39.                                                 0x53,0xe0,0x52,0x10,0x53,0xe0,0x52,0x10,0x73,0xe0,0x12,0x00,0x1e,0x00,0x00,0x00                       
  40.                                                 };

  41. //=========================================================

  42. #ifdef LCD_Work_Mode        //如果定义了,则使用并口工作模式

  43. //=========================================================

  44. /**********************************************************
  45. * 函数功能 ---> LCD12864判断忙
  46. * 入口参数 ---> none
  47. * 返回数值 ---> none
  48. * 功能说明 ---> none
  49. **********************************************************/
  50. void LCD_Check_Busy(void)
  51. {
  52.         static u16 busytimeout;        //忙超时

  53.     //=====================================================
  54.    
  55.     #ifdef LCD_USE_DELAY   /*  定义了则使用延时调整通讯频率    */

  56.     //=====================================================
  57.    
  58.         ktdata = 0xf0;        //数据口全部置位
  59.         do
  60.         {
  61.                 busytimeout++;
  62.                 if(busytimeout > 300)        break;

  63.                 LCD_RS = 0;
  64.                 LCD_Delay();
  65.                 LCD_RW = 1;
  66.                 LCD_Delay();
  67.                 LCD_EN = 1;
  68.                 LCD_Delay();
  69.         }while(ktdata & LCD_Busy == LCD_Busy);        //等待LCD忙完
  70.         LCD_EN = 0;       

  71.     //=====================================================

  72.     #else   /*  没定义则不使用  */

  73.     //=====================================================

  74.     ktdata = 0xf0;        //数据口全部置位
  75.         do
  76.         {
  77.                 busytimeout++;
  78.                 if(busytimeout > 300)        break;

  79.                 LCD_RS = 0;
  80.                 LCD_RW = 1;
  81.                 LCD_EN = 1;
  82. //                LCD_Delay();
  83.         }while(ktdata & LCD_Busy == LCD_Busy);        //等待LCD忙完
  84.         LCD_EN = 0;

  85.     //=====================================================
  86.    
  87.     #endif

  88.     //=====================================================

  89.         if(busytimeout >= 300)        printf("LCD Busy TimeOut...\r\n");
  90.         busytimeout = 0;
  91. }
  92. /**********************************************************
  93. * 函数功能 ---> LCD写入一个字节命令或者数据,判断忙标志(8位数据一次发送完毕)
  94. * 入口参数 ---> dat:要写入的字节
  95. *               ord:判断dat是命令还是数据依据,“0”为命令,“1”为数据
  96. * 返回数值 ---> none
  97. * 功能说明 ---> none
  98. **********************************************************/
  99. void LCD_Write_Byte(u8 dat,u8 ord)
  100. {
  101.     //=====================================================
  102.    
  103.     #ifdef LCD_USE_DELAY   /*  定义了则使用延时调整通讯频率    */

  104.     //=====================================================
  105.    
  106.     LCD_Check_Busy();        //判断是否在忙?
  107.         LCD_EN = 0;
  108.         LCD_Delay();
  109.         LCD_RW = 0;
  110.         LCD_Delay();

  111.         if(ord & 0x01)        LCD_RS = 1;        //放上数据或者命令判断位
  112.         else        LCD_RS = 0;       

  113.         LCD_EN = 1;
  114.         LCD_Delay();
  115.         ktdata = dat;        //放入数据
  116.         LCD_Delay();
  117.         LCD_EN = 0;

  118.     //=====================================================

  119.     #else   /*  没定义则不使用  */

  120.     //=====================================================

  121.     LCD_Check_Busy();        //判断是否在忙?
  122.         LCD_EN = 0;
  123.         LCD_RW = 0;

  124.         if(ord & 0x01)        LCD_RS = 1;        //放上数据或者命令判断位
  125.         else        LCD_RS = 0;       

  126.         LCD_EN = 1;
  127.         ktdata = dat;        //放入数据
  128.         LCD_EN = 0;

  129.     //=====================================================
  130.    
  131.     #endif

  132.     //=====================================================
  133. }

  134. //=========================================================

  135. #else        //没定义则使用串口通讯模式

  136. //=========================================================

  137. /**********************************************************
  138. * 函数功能 ---> LCD启动字节
  139. * 入口参数 ---> dt:传输的字节,高5位在函数内部设定为1
  140. *               只是确定RW和RS为,最后那位也是内部确定
  141. * 返回数值 ---> none
  142. * 功能说明 ---> 首先传送一个启动字节,送入连续5个“1”用来启
  143. *               动一个周期,此时传输计数被重置,并且串行传输
  144. *               被同步,紧接着的两个位指定传输方向(R/W,确
  145. *               定是读还是写)和传输性质(RS,确定是命令寄存
  146. *               器还是数据寄存器),最后的第八位是一个“0”
  147. **********************************************************/
  148. void LCD12864_Start(u8 dt)
  149. {
  150.         u8 dat,j;

  151.         dat = dt | 0xf8;        //高5位设定为“1”,第八位设定为“0”
  152.         LCD_SCL = 0;
  153.         LCD_Delay();LCD_Delay();
  154.         LCD_CS = 1;        //选中显示屏,高电平有效
  155.         for(j = 0;j < 8;j++)
  156.         {
  157.                 if(dat & 0x80)        LCD_SDA = 1;        //放数据到数据线
  158.                 else        LCD_SDA = 0;
  159.                        
  160.                 LCD_Delay();LCD_Delay();LCD_Delay();
  161.                 LCD_SCL = 1;        //发送数据,上升沿有效
  162.                 LCD_Delay();LCD_Delay();LCD_Delay();
  163.                 LCD_SCL = 0;
  164.                 dat <<= 1;        //左移一位,先发的是高位
  165.         }
  166.         LCD_Delay();LCD_Delay();LCD_Delay();
  167.         LCD_SCL = 0;
  168.         LCD_Delay();LCD_Delay();LCD_Delay();        //等待硬件反应(等待数据发送完全)       
  169. }
  170. /**********************************************************
  171. * 函数功能 ---> LCD写入一个字节命令或者数据
  172. * 入口参数 ---> dat:要写入的命令或者数据
  173. *               ord:命令或者数据判断为。0为命令,1为数据
  174. * 返回数值 ---> none
  175. * 功能说明 ---> none
  176. **********************************************************/
  177. void LCD_Write_Byte(u8 dat,u8 ord)
  178. {
  179.         u8 tem;
  180.         u8 i,j;

  181.         tem = dat & 0xf0;        //先发送高4位
  182.         LCD12864_Start((ord << 1) | 0xf0);        //置RW为“0”、RS为“ord”并启动串行传输为数据格式
  183.         for(j = 0;j < 2;j++)        //一个字节数据或者命令分两次发送
  184.         {
  185.                 LCD_SCL = 0;        //允许数据线电平变化
  186.                 LCD_Delay();LCD_Delay();LCD_Delay();

  187.                 for(i = 0;i < 8;i++)
  188.                 {
  189.                         if(tem & 0x80)        LCD_SDA = 1;        //放上数据,屏蔽低4位
  190.                         else        LCD_SDA = 0;       

  191.                         LCD_Delay();LCD_Delay();LCD_Delay();
  192.                         LCD_SCL = 1;        //发送数据,上升沿有效
  193.                         LCD_Delay();LCD_Delay();LCD_Delay();
  194.                         LCD_SCL = 0;
  195.                         tem <<= 1;        //左移一位,先发的是高位
  196.                 }
  197.                 tem = (dat << 4) & 0xf0;        //发完高4位,再次发送低4位
  198.         }
  199.         LCD_Delay();LCD_Delay();LCD_Delay();
  200.         LCD_SDA = 0;
  201.         LCD_Delay();LCD_Delay();LCD_Delay();
  202.         LCD_SCL = 0;
  203.         LCD_Delay();LCD_Delay();LCD_Delay();
  204.         LCD_CS = 0;
  205.         LCD_Delay();LCD_Delay();LCD_Delay();        //等待硬件反应(等待数据发送完全)
  206. }

  207. //=========================================================

  208. #endif

  209. //=========================================================

  210. /**********************************************************
  211. * 函数功能 ---> 设置LCD显示位置
  212. * 入口参数 ---> x:行,取值范围:1 ~ 4
  213. *               y:列,取值范围:0 ~ 7
  214. * 返回数值 ---> none
  215. * 功能说明 ---> none
  216. **********************************************************/
  217. void LCD_Set_xy(u8 x,u8 y)
  218. {
  219.         switch(x)
  220.         {
  221.                 case 1:
  222.                                         LCD_Write_Byte((One_LineAddress + y),0);        //写入操作地址
  223.                                         break;
  224.                 case 2:
  225.                                         LCD_Write_Byte((Two_LineAddress + y),0);        //写入操作地址
  226.                                         break;
  227.                 case 3:
  228.                                         LCD_Write_Byte((Three_LineAddress + y),0);        //写入操作地址
  229.                                         break;
  230.                 case 4:
  231.                                         LCD_Write_Byte((Four_LineAddress + y),0);        //写入操作地址
  232.                                         break;
  233.                 default:
  234.                                         LCD_Write_Byte((One_LineAddress + y),0);        //写入操作地址
  235.                                         break;
  236.         }
  237. }
  238. /**********************************************************
  239. * 函数功能 ---> 写入自定义字符到LCD的CGRAM
  240. * 入口参数 ---> *data_code:写入的数组
  241. * 返回数值 ---> none
  242. * 功能说明 ---> none
  243. **********************************************************/
  244. void LCD_Write_CGRAM(u8 *data_code)
  245. {
  246.         u8 i;
  247.         LCD_Write_Byte(0x34,0);        //打开字符扩展指令
  248.         LCD_Write_Byte(0x02,0);        //SR=0,允许输入
  249.         LCD_Write_Byte(0x30,0);        //恢复基本指令
  250.         LCD_Write_Byte(0x40,0);        //CGRAM地址
  251.         for(i = 0;i < CGRAM_Value;i++)
  252.         {
  253.                 LCD_Write_Byte(data_code[i*2],1);
  254.                 LCD_Write_Byte(data_code[i*2+1],1);
  255.         }
  256. }
  257. /**********************************************************
  258. * 函数功能 ---> 显示一个CGRAM内容
  259. * 入口参数 ---> x:行
  260. *               y:列
  261. *               *str为要显示的字符串
  262. *               add_h:CGRAM高位地址
  263. *               add_l:CGRAM低位地址
  264. * 返回数值 ---> none
  265. * 功能说明 ---> none
  266. **********************************************************/
  267. void Display_CGRAM(u8 x,u8 y,u8 add_h,u8 add_l)
  268. {
  269.         LCD_Set_xy(x,y);        //设置显示地址
  270.         LCD_Write_Byte(add_h,1);
  271.         LCD_Write_Byte(add_l,1);
  272. }
  273. /**********************************************************
  274. * 函数功能 ---> LCD显示字符串
  275. * 入口参数 ---> x:行
  276. *               y:列
  277. *               *str:要显示的字符串
  278. *               len:显示的长度
  279. * 返回数值 ---> none
  280. * 功能说明 ---> 只能从数组的开头开始显示len长度字符串
  281. **********************************************************/
  282. void Display_String(u8 x,u8 y,u8 *str,u8 len)
  283. {
  284.         LCD_Set_xy(x,y);        //设置显示地址
  285.         while(len--)
  286.         {
  287.                 LCD_Write_Byte(*str,1);
  288.                 str++;
  289.         }
  290. }
  291. /**********************************************************
  292. * 函数功能 ---> LCD显示字符串
  293. * 入口参数 ---> x:行
  294. *               y:列
  295. *               *str:要显示的字符串
  296. *               len:显示的长度
  297. * 返回数值 ---> none
  298. * 功能说明 ---> 可以从显示数组的任意位置starsum开始显示len长度的字符串
  299. **********************************************************/
  300. void Display_String1(u8 x,u8 y,u8 *str,u8 starsum,u8 len)
  301. {
  302.         u8 i;

  303.         LCD_Set_xy(x,y);        //设置显示地址

  304.         for(i = starsum;i < starsum + len;i++)
  305.         {
  306.                 LCD_Write_Byte(str[i],1);
  307.         }
  308. }
  309. /**********************************************************
  310. * 函数功能 ---> LCD写入GDRAM绘图
  311. * 入口参数 ---> x:行
  312. *               y:列
  313. *               width:显示图片的宽度
  314. *               hieght:显示图片的高度
  315. *               *bmp:要显示的字符串
  316. *               dis_mode:显示模式,是半屏显示还是全屏显示
  317. *                         0:半屏显示。1:全屏显示
  318. * 返回数值 ---> none
  319. * 功能说明 ---> none
  320. **********************************************************/
  321. void Display_GDRAM(u8 x,u8 y,u8 width,u8 height,u8 *bmp,u8 dis_mode)
  322. {
  323.         u8 i,j,k;
  324.         u8 base_x,base_y;        //起始坐标

  325.         /******全屏绘图显示******/
  326.         if(dis_mode)
  327.         {
  328.                 switch(x)
  329.                 {
  330.                         case 1:
  331.                                         base_y = One_LineAddress + y;
  332.                                         break;
  333.                         case 2:
  334.                                         base_y = Two_LineAddress + y;
  335.                                         break;
  336.                         case 3:
  337.                                         base_y = Three_LineAddress + y;
  338.                                         break;
  339.                         case 4:
  340.                                         base_y = Four_LineAddress + y;
  341.                                         break;
  342.                         default:        break;
  343.                 }
  344.                  
  345.                  LCD_Write_Byte(MPU_8bit_Expansion_Draw_Close,0);        //扩充指令
  346.                 LCD_Write_Byte(MPU_8bit_Expansion_Draw_Open,0);        //打开绘图功能
  347.                
  348.                 for(j = 0;j < height;j++)        //32
  349.                 {
  350.                         LCD_Write_Byte(base_y + j,0);        //写入行号,即第几行开始
  351.                         LCD_Write_Byte(One_LineAddress,0);        //横坐标的第几个字节开始写
  352.                        
  353.                         for(i = 0;i < width;i++)        //上半屏
  354.                         {
  355.                                 LCD_Write_Byte(bmp[width * j + i],1);        //开始写入数据
  356.                         }
  357.                         for(k = 0;k < width;k++)        //下半屏
  358.                         {
  359.                                  LCD_Write_Byte(bmp[width * (j + height) + k],1);        //开始写入数据
  360.                         }
  361.                 }                
  362.         }
  363.         /******分开上半屏或者下半屏写******/
  364.         else
  365.         {
  366.                 switch(x)        //判断显示的横坐标,确定起始行
  367.                 {
  368.                         case 1:        //上半屏
  369.                                         base_x = One_LineAddress;        //起始地址为0x80
  370.                                         break;
  371.        
  372.                         case 2:        //下半屏
  373.                                         base_x = One_LineAddress;        //起始地址为0x80
  374.                                         break;
  375.        
  376.                         case 3:        //下半屏
  377.                                         base_x = Three_LineAddress;        //起始地址为0x88
  378.                                         break;
  379.                         default:
  380.                                         break;
  381.                 }
  382.                
  383.                 base_y = base_x + y;        //x轴方向的偏移,基地址 + 偏移地址
  384.                
  385.                 LCD_Write_Byte(MPU_8bit_Expansion_Draw_Close,0);        //扩充指令
  386.                 LCD_Write_Byte(MPU_8bit_Expansion_Draw_Open,0);        //打开绘图功能
  387.                
  388.                 if(x == 1 || x == 3)        //直接在同一个半屏的,直接写完32位数据
  389.                 {
  390.                         for(i = 0;i < 32;i++)        //写地址有讲究,先写垂直再写水平,详见“设定绘图RAM地址”说明
  391.                         {
  392.                                 LCD_Write_Byte(One_LineAddress + i,0);        //写入垂直地址,不管上半屏还是下半屏
  393.                                 LCD_Write_Byte(base_y,0);        //写入水平地址
  394.                                
  395.                                 for(j = 0;j < 2;j++)        //两组8bit数据,组成16bit
  396.                                 {
  397.                                         LCD_Write_Byte(*bmp++,1);        //写入数据
  398.                                 }
  399.                         }
  400.                 }
  401.                
  402.                 if(x == 2)        //数据显示不在同一半屏上的,先写上半屏,在写下半屏
  403.                 {
  404.                         for(i = 0;i < 16;i++)        //上半屏16行点阵数据
  405.                         {
  406.                                 LCD_Write_Byte(Two_LineAddress + i,0);        //写入垂直地址
  407.                                 LCD_Write_Byte(base_y,0);        //写入水平地址
  408.                                
  409.                                 for(j = 0;j < 2;j++)        //两组8bit数据,组成16bit
  410.                                 {
  411.                                         LCD_Write_Byte(*bmp++,1);        //写入数据
  412.                                 }
  413.                         }
  414.                         for(i = 0;i < 16;i++)        //下半屏16行点阵数据
  415.                         {
  416.                                 LCD_Write_Byte(Three_LineAddress + i,0);        //写入垂直地址
  417.                                 LCD_Write_Byte(base_y + 8,0);        //写入水平地址
  418.                                
  419.                                 for(j = 0;j < 2;j++)        //两组8bit数据,组成16bit
  420.                                 {
  421.                                         LCD_Write_Byte(*bmp++,1);        //写入数据
  422. ……………………

  423. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

所有资料51hei提供下载:

STM32F103VCT612864.rar (263.89 KB, 下载次数: 100)
lcd12864.zip (8.03 KB, 下载次数: 83)
JN12864J使用说明书.pdf (776.12 KB, 下载次数: 40)




评分

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

查看全部评分

回复

使用道具 举报

ID:44037 发表于 2017-7-21 19:19 | 显示全部楼层
谢谢楼主分享  学习一下  现在能力只能模仿  无力独立完成
回复

使用道具 举报

ID:329099 发表于 2018-5-13 20:16 | 显示全部楼层
谢谢楼主分享  学习一下  现在能力只能模仿  无力独立完成
回复

使用道具 举报

ID:424360 发表于 2018-11-25 11:53 | 显示全部楼层
谢谢楼主分享  学习一下  现在能力只能模仿  无力独立完成
回复

使用道具 举报

ID:388459 发表于 2018-11-26 11:25 | 显示全部楼层
谢谢分享  学习一下
回复

使用道具 举报

ID:100415 发表于 2019-7-29 14:16 | 显示全部楼层
谢谢楼主分享,新人学习一下
回复

使用道具 举报

ID:100415 发表于 2019-7-29 14:16 | 显示全部楼层
谢谢楼主分享,学习一下
回复

使用道具 举报

ID:338360 发表于 2020-4-7 00:00 | 显示全部楼层
谢谢!刚好需要,下来参考一下
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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