15
2020
10

CT107D DS1302

使用官方的ds1302.h和ds1302.c,下方有核心代码讲解

main.c

#include<reg52.h>
#include<intrins.h>
#include<ds1302.h>
typedef unsigned char BYTE;
typedef unsigned int WORD;

#define FOSC 11059200L          //系统频率
#define BAUD 115200             //串口波特率

#define NONE_PARITY     0       //无校验
#define ODD_PARITY      1       //奇校验
#define EVEN_PARITY     2       //偶校验
#define MARK_PARITY     3       //标记校验
#define SPACE_PARITY    4       //空白校验

#define PARITYBIT EVEN_PARITY   //定义校验位
#define uchar unsigned char   //定义校验位

unsigned char code ds[] = {55,59,23,10,10,1,16};
//秒  分钟  小时  日期  月份  星期 年份

sfr P0M1 = 0x93;
sfr P0M0 = 0x94;
sfr P1M1 = 0x91;
sfr P1M0 = 0x92;
sfr P2M1 = 0x95;
sfr P2M0 = 0x96;
sfr P3M1 = 0xb1;
sfr P3M0 = 0xb2;
sfr P4M1 = 0xb3;
sfr P4M0 = 0xb4;
sfr P5M1 = 0xC9;
sfr P5M0 = 0xCA;
sfr P6M1 = 0xCB;
sfr P6M0 = 0xCC;
sfr P7M1 = 0xE1;
sfr P7M0 = 0xE2;

sfr AUXR  = 0x8e;               //辅助寄存器

sfr P_SW1   = 0xA2;             //外设功能切换寄存器1

#define S1_S0 0x40              //P_SW1.6
#define S1_S1 0x80              //P_SW1.7

sbit P22 = P2^2;

bit busy;

void SendData(BYTE dat);
void SendString(char *s);
void uart_init();
void settime();
void hdw_init();
void sendtime();
//*************************************
void main()
{
	hdw_init();
	settime();
	uart_init();
	while(1)
	{
		sendtime();
	}
}
//*************************************
void sendtime()
{
	unsigned char i;
	unsigned adr=0x81;
	unsigned char now[7]={0};
	for(i=0;i<7;i++)
	{
		 now[i]=Read_Ds1302_Byte(adr+i);
		 adr+=2;
	}
	SendString("\nThe time now is:");
	SendData('0'+now[0]/16);
	SendData('0'+now[0]%16);
	SendString("sec");
	SendData('0'+now[1]/16);
	SendData('0'+now[1]%16);
	SendString("min");
	SendData('0'+now[2]/16);
	SendData('0'+now[2]%16);
	SendString("Hour");
	SendData('0'+now[3]/16);
	SendData('0'+now[3]%16);
	SendString("Data");
	SendData('0'+now[4]/16);
	SendData('0'+now[4]%16);
	SendString("Month");
	SendData('0'+now[6]/16);
	SendData('0'+now[6]%16);
	SendString("Year");
}
void uart_init()
{
	P0M0 = 0x00;
    P0M1 = 0x00;
    P1M0 = 0x00;
    P1M1 = 0x00;
    P2M0 = 0x00;
    P2M1 = 0x00;
    P3M0 = 0x00;
    P3M1 = 0x00;
    P4M0 = 0x00;
    P4M1 = 0x00;
    P5M0 = 0x00;
    P5M1 = 0x00;
    P6M0 = 0x00;
    P6M1 = 0x00;
    P7M0 = 0x00;
    P7M1 = 0x00;

    ACC = P_SW1;
    ACC &= ~(S1_S0 | S1_S1);    //S1_S0=0 S1_S1=0
    P_SW1 = ACC;                //(P3.0/RxD, P3.1/TxD)
    
//  ACC = P_SW1;
//  ACC &= ~(S1_S0 | S1_S1);    //S1_S0=1 S1_S1=0
//  ACC |= S1_S0;               //(P3.6/RxD_2, P3.7/TxD_2)
//  P_SW1 = ACC;  
//  
//  ACC = P_SW1;
//  ACC &= ~(S1_S0 | S1_S1);    //S1_S0=0 S1_S1=1
//  ACC |= S1_S1;               //(P1.6/RxD_3, P1.7/TxD_3)
//  P_SW1 = ACC;  

#if (PARITYBIT == NONE_PARITY)
    SCON = 0x50;                //8位可变波特率
#elif (PARITYBIT == ODD_PARITY) || (PARITYBIT == EVEN_PARITY) || (PARITYBIT == MARK_PARITY)
    SCON = 0xda;                //9位可变波特率,校验位初始为1
#elif (PARITYBIT == SPACE_PARITY)
    SCON = 0xd2;                //9位可变波特率,校验位初始为0
#endif

    AUXR = 0x40;                //定时器1为1T模式
    TMOD = 0x00;                //定时器1为模式0(16位自动重载)
    TL1 = (65536 - (FOSC/4/BAUD));   //设置波特率重装值
    TH1 = (65536 - (FOSC/4/BAUD))>>8;
    TR1 = 1;                    //定时器1开始启动
    ES = 1;                     //使能串口中断
    EA = 1;
}
void hdw_init()
{
	P2=0xA0;
	P0=0x00;
	P2=0x00;
	
	P2=0x80;
	P0=0xFF;
	P2=0x00;
}
void settime()
{
    uchar i;
    uchar add = 0x80;
    Write_Ds1302_Byte(0x8e,0x00);
    for(i = 0;i < 7;i++)
    {
        Write_Ds1302_Byte(add,ds[i]);
        add = add+2;
    }
    Write_Ds1302_Byte (0x8e, 0x80);
}
/*----------------------------
UART 中断服务程序
-----------------------------*/
void Uart() interrupt 4
{
    if (RI)
    {
        RI = 0;                 //清除RI位
        P0 = SBUF;              //P0显示串口数据
        P22 = RB8;              //P2.2显示校验位
    }
    if (TI)
    {
        TI = 0;                 //清除TI位
        busy = 0;               //清忙标志
    }
}

/*----------------------------
发送串口数据
----------------------------*/
void SendData(BYTE dat)
{
    while (busy);               //等待前面的数据发送完成
    ACC = dat;                  //获取校验位P (PSW.0)
    if (P)                      //根据P来设置校验位
    {
#if (PARITYBIT == ODD_PARITY)
        TB8 = 0;                //设置校验位为0
#elif (PARITYBIT == EVEN_PARITY)
        TB8 = 1;                //设置校验位为1
#endif
    }
    else
    {
#if (PARITYBIT == ODD_PARITY)
        TB8 = 1;                //设置校验位为1
#elif (PARITYBIT == EVEN_PARITY)
        TB8 = 0;                //设置校验位为0
#endif
    }
    busy = 1;
    SBUF = ACC;                 //写数据到UART数据寄存器
}

/*----------------------------
发送字符串
----------------------------*/
void SendString(char *s)
{
    while (*s)                  //检测字符串结束标志
    {
        SendData(*s++);         //发送当前字符
    }
}

设置初始时间

void settime()
{
    uchar i;
    uchar add = 0x80;
    Write_Ds1302_Byte(0x8e,0x00);
    for(i = 0;i < 7;i++)
    {
        Write_Ds1302_Byte(add,ds[i]);
        add = add+2;
    }
    Write_Ds1302_Byte (0x8e, 0x80);
}

为什么要向0x8e这个地址送去0x00呢?

Write_Ds1302_Byte(0x8e,0x00);

这是因为DS1302这个芯片

image.png

在write写入0x8Eh可以控制数据写入,WP的意思是wirte protect,如果WP为0,则数据无法被写入到芯片,需要将该位置0。

for(i = 0;i < 7;i++)
    {
        Write_Ds1302_Byte(add,ds[i]);
        add = add+2;
    }

那么为什么要使用for循环去写入数据呢?就是为了方便写入数据,实际上我们将ds数组中的值分别送进了DS1302从0x80-0x8C的地址内,对应图中的

image.png

它们的低四位是个位的值,高八位是十位数字。

而在最后的

Write_Ds1302_Byte (0x8e, 0x80);

其实和第一个的

Write_Ds1302_Byte(0x8e,0x00);

类似,就是打开芯片的WP,就是写入保护,此时初始化已经完成,不能再写入数据了。


接下来看怎么来读取时间?我们通过上面的图可以类似得到,我们想得到现在的时间,只需要读取对应的地址即可

image.png

unsigned char i;
	unsigned adr=0x81;
	unsigned char now[7]={0};
	for(i=0;i<7;i++)
	{
		 now[i]=Read_Ds1302_Byte(adr+i);
		 adr+=2;
	}


« 上一篇 下一篇 »

返回顶部
请先 登录 再评论,若不是会员请先 注册