LCD1602液晶显示DS1302时钟

主要是熟悉下LCD1602和DS1302模块。关于DS1302更具体的在上一篇文章《DS1302驱动》。

main.c

#include<reg52.h> //包含头文件,一般情况不需要改动,头文件包含特殊功能寄存器的定义
#include <stdio.h>
#include "ds1302.h"
#include "delay.h"
#include "1602.h"

bit ReadTimeFlag;//定义读时间标志
bit SetFlag;     //更新时间标志位
unsigned char time_buf2[16];  // 用于临时存储从串口接收的时间数据

void Init_Timer0(void);//定时器初始化
void UART_Init(void);
/*------------------------------------------------
                    主函数
------------------------------------------------*/
void main (void)
{
unsigned char i;                  
unsigned char temp[16];//定义显示区域临时存储数组

LCD_Init();           //初始化液晶
DelayMs(20);          //延时有助于稳定
LCD_Clear();          //清屏
Init_Timer0();        //定时器0初始化
Ds1302_Init();        //ds1302初始化
UART_Init();          //串口初始化

Ds1302_Read_Time();   //首次读取时间
if((time_buf1[2]+time_buf1[7])==0) //如果所有参数都为0,写入一个初始值
   Ds1302_Write_Time();    
while (1)         //主循环
  {


if(SetFlag)     //如果接收到串口信息则更新时钟
  {
	for(i=0;i<8;i++)
		{
		time_buf1[i]=time_buf2[2*i]*10+time_buf2[2*i+1];//数据整合,如2个数 1和5整合成15
		}
		Ds1302_Write_Time();//接收更新的时间然后写入ds1302
		SetFlag=0;          //时钟信息更新后标志位清零
   }


if(ReadTimeFlag==1) //定时读取ds1302 定时时间到 则标志位置1,处理过时间参数标志位清零
{
  ReadTimeFlag=0;  //标志位清零
  Ds1302_Read_Time();//读取时间参数
  sprintf(temp,"DATE %02d-%02d-%02d %d",(int)time_buf1[1],(int)time_buf1[2],(int)time_buf1[3],(int)time_buf1[7]);//年月日周
  LCD_Write_String(0,0,temp);//显示第一行
  sprintf(temp,"TIME %02d:%02d:%02d",(int)time_buf1[4],(int)time_buf1[5],(int)time_buf1[6]);//时分秒
  LCD_Write_String(0,1,temp);//显示第二行
  }
 }
}
/*------------------------------------------------
              串口通讯初始化
------------------------------------------------*/
void UART_Init(void)
{
    SCON  = 0x50;		        // SCON: 模式 1, 8-bit UART, 使能接收  
    TMOD |= 0x20;               // TMOD: timer 1, mode 2, 8-bit 重装
    TH1   = 0xFD;               // TH1:  重装值 9600 波特率 晶振 11.0592MHz  
    TR1   = 1;                  // TR1:  timer 1 打开                         
    EA    = 1;                  //打开总中断
    ES    = 1;                  //打开串口中断
}

/*------------------------------------------------
                    定时器初始化子程序
------------------------------------------------*/
void Init_Timer0(void)
{
 TMOD |= 0x01;	  //使用模式1,16位定时器,使用"|"符号可以在使用多个定时器时不受影响		     
 //TH0=0x00;	      //给定初值
 //TL0=0x00;
 EA=1;            //总中断打开
 ET0=1;           //定时器中断打开
 TR0=1;           //定时器开关打开
}
/*------------------------------------------------
                 定时器中断子程序
------------------------------------------------*/
void Timer0_isr(void) interrupt 1 
{
 static unsigned int num;
 TH0=(65536-2000)/256;		  //重新赋值 2ms
 TL0=(65536-2000)%256;
 
 num++;
 if(num==50)        //大致100ms
   {
    num=0;
    ReadTimeFlag=1; //读标志位置1
	}
}

/*------------------------------------------------
                  串口中断程序
------------------------------------------------*/
void UART_SER (void) interrupt 4 //串行中断服务程序
{
    unsigned char Temp;          //定义临时变量   用于存储串口接收到的数据
    unsigned char i;   //用于存储接收到的字符数量
    if(RI)                        //判断是接收中断产生
     {
	  RI=0;                      //标志位清零
	  Temp=SBUF;                 //读入缓冲区的值  从串口数据缓冲区 SBUF 中读取接收到的数据
	  time_buf2[i]=Temp&0x0F;    //将接收到的数据的低四位存入 time_buf2 数组中,可能用于特定的数据处理或解析
	  i++;      //增接收计数器,用于记录接收到的字符数量
	  if(i==16)                  //连续接收16个字符信息
	   {
	    i=0;
		SetFlag=1;               //接收完成标志位置1
	   }
      SBUF=Temp; //把接收到的值再发回电脑端
	 }
   if(TI)  //如果是发送标志位,清零
     TI=0;
} 

1602.c

#include "1602.h"
#include "delay.h"

sbit RS = P2^4;   //定义端口   寄存器选择
sbit RW = P2^5; //读写控制
sbit EN = P2^6; //使能信号

#define RS_CLR RS=0 //当 RS 被设置为 0 时,表示准备发送的是命令
#define RS_SET RS=1 //当 RS 被设置为 1 时,表示准备发送的是数据

#define RW_CLR RW=0 // RW 被设置为 0 时,表示写入操作  如显示字符、清除屏幕等。通常情况下,液晶显示屏的大多数操作都涉及写入操作,因此 RW 经常设置为 0。
#define RW_SET RW=1 //RW 被设置为 1 时,表示读取操作   如检查LCD是否忙碌(busy flag)、读取显示数据等。读取操作是比较少见的,通常用于特定的状态检查或调试。

#define EN_CLR EN=0 //将 EN 设置为 0 表示不使能,LCD 控制器处于待机状态,不会进行任何操作
#define EN_SET EN=1 //将 EN 设置为 1 表示使能,LCD 控制器会执行之前设置好的命令或数据写入操作

#define DataPort P0

/*------------------------------------------------
              判忙函数  检查LCD是否忙碌的函数。在液晶屏忙于执行命令时返回1,否则返回0   内部的顺序就是基本上都是固定的
------------------------------------------------*/
 bit LCD_Check_Busy(void) 
 { 
 DataPort= 0xFF; //将 DataPort 设置为 0xFF 的目的是将 DataPort 上的所有数据线(8根数据线)均设置为逻辑高电平(1)。这样做的原因是在 LCD 控制中,首先需要确保数据线上的数据为一个已知状态。通过将 DataPort 设置为 0xFF,可以清除数据线上的任何潜在残留数据,确保准备发送或接收新的数据时,数据线处于初始状态。所以也可以设置为0X00。
 RS_CLR; 
 RW_SET; //检查LCD是否忙碌
 EN_CLR;  //将 EN 引脚清零 (EN_CLR;)。这一步通常用于确保之前的数据传输完全结束,或者在开始新的数据传输之前,确保 LCD 控制器处于稳定状态。通过清零 EN 引脚,可以避免不必要的干扰或误操作。
 _nop_(); // 是一个空指令,用于在程序中插入一个小的延迟。这种延迟可以帮助确保 EN 引脚的状态变化被正确地读取和响应。在液晶显示屏的控制中,确保时序的正确性是非常关键的,特别是在高速操作中。
 EN_SET; //将 EN 引脚置为高电平 (EN_SET;),触发 LCD 控制器执行之前设置好的命令或数据传输。EN 设置为高电平时,LCD 控制器会根据之前设置好的 RS 和 RW 引脚的状态执行相应的操作。
 return (bit)(DataPort & 0x80);//DataPort的最高位是否为1,若为1则返回1,否则返回0
 }
/*------------------------------------------------
              写入命令函数
------------------------------------------------*/
 void LCD_Write_Com(unsigned char com) 
 {  
 while(LCD_Check_Busy()); //忙则等待
 RS_CLR; //告诉LCD要发送的是命令
 RW_CLR; //要进行写操作
 EN_SET; //进行上面的操作
 DataPort= com; //发送的命令数据 com 放入 DataPort(数据端口)
 _nop_(); //空指令,用来确保时序的正确性
 EN_CLR;// 结束命令
 }
/*------------------------------------------------
              写入数据函数
------------------------------------------------*/
 void LCD_Write_Data(unsigned char Data) 
 { 
 while(LCD_Check_Busy()); //忙则等待
 RS_SET; 
 RW_CLR; 
 EN_SET; 
 DataPort= Data; 
 _nop_();
 EN_CLR;
 }

/*------------------------------------------------
                清屏函数
------------------------------------------------*/
 void LCD_Clear(void) 
 { 
 LCD_Write_Com(0x01); //向 LCD1602 写入命令 0x01是清屏操作,并将光标返回到起始位置(第一行第一列)
 DelayMs(5);
 }
/*------------------------------------------------
              写入字符串函数
------------------------------------------------*/
 void LCD_Write_String(unsigned char x,unsigned char y,unsigned char *s) //x  哪一列  y哪一行 s字符串 
 {     
 if (y == 0)   //LCD1602总共就两行   
 	{     
	 LCD_Write_Com(0x80 + x);     //表示第一行       第一行:从0x80到0x8F    
 	}
 else 
 	{      
 	LCD_Write_Com(0xC0 + x);      //表示第二行       第二行:从0xC0到0xCF
 	}        
 while (*s) 
 	{     
 LCD_Write_Data( *s);     
 s ++;     
 	}
 }
/*------------------------------------------------
              写入字符函数
------------------------------------------------*/
/* void LCD_Write_Char(unsigned char x,unsigned char y,unsigned char Data) 
 {     
 if (y == 0) 
 	{     
 	LCD_Write_Com(0x80 + x);     
 	}    
 else 
 	{     
 	LCD_Write_Com(0xC0 + x);     
 	}        
 LCD_Write_Data( Data);  
 }*/
/*------------------------------------------------
              初始化函数
------------------------------------------------*/
 void LCD_Init(void) 
 {
   LCD_Write_Com(0x38);    /*显示模式设置   0x38 是设置显示模式的命令,通常用于设置显示为两行、5x8 点阵的字符 */ 
   DelayMs(5); 
   LCD_Write_Com(0x38); 
   DelayMs(5); 
   LCD_Write_Com(0x38); 
   DelayMs(5); 
   LCD_Write_Com(0x38);  
   LCD_Write_Com(0x08);    /*显示关闭  将显示屏关闭,但并不清除显示内容*/   
   LCD_Write_Com(0x01);    /*显示清屏*/ 
   LCD_Write_Com(0x06);    /*显示光标移动设置   设置光标的移动方向和显示的滚动方式*/ 
   DelayMs(5); 
   LCD_Write_Com(0x0C);    /*显示开及光标设置   打开显示、关闭光标闪烁的命令,以正常显示字符内容*/
   }
   

DS1302.c

unsigned char time_buf1[8] = {20,10,6,5,12,55,00,6};//空年月日时分秒周
unsigned char time_buf[8] ;                         //空年月日时分秒周
/*------------------------------------------------
           向DS1302写入一字节数据
------------------------------------------------*/
void Ds1302_Write_Byte(unsigned char addr, unsigned char d)
{

	unsigned char i;
	RST_SET;	
	
	//写入目标地址:addr
	addr = addr & 0xFE;     //最低位置零
	for (i = 0; i < 8; i ++) 
	    { 
		if (addr & 0x01) 
		    {
			IO_SET;
			}
		else 
		    {
			IO_CLR;
			}
		SCK_SET;
		SCK_CLR;
		addr = addr >> 1;
		}
	
	//写入数据:d
	for (i = 0; i < 8; i ++) 
	   {
		if (d & 0x01) 
		    {
			IO_SET;
			}
		else 
		    {
			IO_CLR;
			}
		SCK_SET;
		SCK_CLR;
		d = d >> 1;
		}
	RST_CLR;					//停止DS1302总线
}
/*------------------------------------------------
           从DS1302读出一字节数据
------------------------------------------------*/

unsigned char Ds1302_Read_Byte(unsigned char addr) 
{

	unsigned char i;
	unsigned char temp;
	RST_SET;	

	//写入目标地址:addr
	addr = addr | 0x01;//最低位置高
	for (i = 0; i < 8; i ++) 
	    {
	     
		if (addr & 0x01) 
		   {
			IO_SET;
			}
		else 
		    {
			IO_CLR;
			}
		SCK_SET;
		SCK_CLR;
		addr = addr >> 1;
		}
	
	//输出数据:temp
	for (i = 0; i < 8; i ++) 
	    {
		temp = temp >> 1;
		if (IO_R) 
		   {
			temp |= 0x80;
			}
		else 
		   {
			temp &= 0x7F;
			}
		SCK_SET;
		SCK_CLR;
		}
	
	RST_CLR;	//停止DS1302总线
	return temp;
}

/*------------------------------------------------
           向DS1302写入时钟数据
------------------------------------------------*/
void Ds1302_Write_Time(void) 
{
     
    unsigned char i,tmp;
	for(i=0;i<8;i++)
	    {                  //BCD处理
		tmp=time_buf1[i]/10;
		time_buf[i]=time_buf1[i]%10;
		time_buf[i]=time_buf[i]+tmp*16;
	    }
	Ds1302_Write_Byte(ds1302_control_add,0x00);			//关闭写保护 
	Ds1302_Write_Byte(ds1302_sec_add,0x80);				//暂停 
	//Ds1302_Write_Byte(ds1302_charger_add,0xa9);			//涓流充电 
	Ds1302_Write_Byte(ds1302_year_add,time_buf[1]);		//年 
	Ds1302_Write_Byte(ds1302_month_add,time_buf[2]);	//月 
	Ds1302_Write_Byte(ds1302_date_add,time_buf[3]);		//日 
	Ds1302_Write_Byte(ds1302_day_add,time_buf[7]);		//周 
	Ds1302_Write_Byte(ds1302_hr_add,time_buf[4]);		//时 
	Ds1302_Write_Byte(ds1302_min_add,time_buf[5]);		//分
	Ds1302_Write_Byte(ds1302_sec_add,time_buf[6]);		//秒
	Ds1302_Write_Byte(ds1302_day_add,time_buf[7]);		//周 
	Ds1302_Write_Byte(ds1302_control_add,0x80);			//打开写保护 
}

/*------------------------------------------------
           从DS1302读出时钟数据
------------------------------------------------*/
void Ds1302_Read_Time(void)  
{ 
   	    unsigned char i,tmp;
	time_buf[1]=Ds1302_Read_Byte(ds1302_year_add);		//年 
	time_buf[2]=Ds1302_Read_Byte(ds1302_month_add);		//月 
	time_buf[3]=Ds1302_Read_Byte(ds1302_date_add);		//日 
	time_buf[4]=Ds1302_Read_Byte(ds1302_hr_add);		//时 
	time_buf[5]=Ds1302_Read_Byte(ds1302_min_add);		//分 
	time_buf[6]=(Ds1302_Read_Byte(ds1302_sec_add))&0x7F;//秒 
	time_buf[7]=Ds1302_Read_Byte(ds1302_day_add);		//周 


	for(i=0;i<8;i++)
	   {           //BCD处理
		tmp=time_buf[i]/16;
		time_buf1[i]=time_buf[i]%16;
		time_buf1[i]=time_buf1[i]+tmp*10;
	   }
}

/*------------------------------------------------
                DS1302初始化
------------------------------------------------*/
void Ds1302_Init(void)
{
	
	RST_CLR;			//RST脚置低
	SCK_CLR;			//SCK脚置低
    Ds1302_Write_Byte(ds1302_sec_add,0x00);				 
}

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值