主要是熟悉下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);
}