STC实验箱4 IAP15W4K58S4 Keil uVision V5.29.0.0 PK51 Prof.Developers Kit Version:9.60.0.0
硬知识
???????摘自《PCF8563实时时钟日历芯片选型指南》。
概述
???????PCF8563是低功耗的CMOS实时时钟日历芯片。它提供一个可编程时钟输出一个中断输出和掉电检测器所有的地址和数据,通过12C总线接口串行传递最大总线速度为400Kbits/s,每次读写数据后内嵌的字地址寄存器会自动产生增量。
特性
低工作电流典型值:0.25 A(VDD=3.0V
T
a
m
b
T_{amb}
Tamb?=25) 世纪标志; 大工作电压范围:1.0V–5.5V; 低休眠电流典型值为:0.25 A(VDD=3.0V,
T
a
m
b
T_{amb}
Tamb?=25) 400KHz 的I2C接口:(VDD=1.8~5.5V) 可编程时钟输出频率为:32.768KHz,1024Hz、32Hz,1Hz; 报警和定时器; 内部集成的振荡器、电容、片内电源复位功能掉电检测器; I2C 从地址:读0A3H; 写0A2H; 开漏中断引脚。
功能描述
???????PCF8563内有16个8位的地址递增寄存器,一个32.768 kHz片上集成电容振荡器,一个实时时钟源(RTC)的分频器,可编程的时钟输出,一个定时器,报警器,一个低压检测器和400KHz的I2C接口。 ???????所有16个寄存器被设计成可寻址的8位并行寄存器,虽然不是所有的位都有效。前两个寄存器(内存地址00H和01H),用于控制与或状态寄存器。内存地址02H至08H是时钟功能的计数器,用于(秒、分、时、日、月、年计数器),内存地址09H至0CH包含定义报警的条件的报警寄存器。内存地址0DH控制CLKOUT的输出频率.0EH和0FH分别是定时控制器和定时器。 ???????秒、分钟、小时、天、月、年、以及每分钟报警、小时报警、日报警寄存器都以BCD格式编码。平日和星期报警寄存器不以BCD格式编码。 ???????当一个RTC寄存器被读取,所有的寄存器的内容被冻结。因此可以避免在读指令跳转期间,读取时钟/日历时发生错误。
报警功能模式
???????通过清除一个或多个报警寄存器最高有效位(位AE-报警启用),相应的报警条件将被激活。这种方式可以产生从每分钟至每周一次的报警。报警条件设置报警标志,AF(控制/状态寄存器2的第3位),AF可用于产生一个中断(INT),AF只能通过软件清零。
定时器模式
???????8位减数计时器(地址0FH)由定时控制寄存器(地址0EH,参见表25)控制。定时控制寄存器可以选择定时器的时钟源频率(4096,64,1,或1/60Hz)和启用/禁用计时器。从软件加载的8位二进制值的倒计时,在每个倒计时结束时,定时器设置的定时器标志TF(见表7)。定时器标志位TF只能由软件清零。根据定时器标志位TF可以产生一个中断(INT)。每个倒计时阶段都可能会产生中断脉冲信号,作为一个永久的积极信号,如TF条件下。TI/TP(见表7)用于控制这种模式的选择。当读取定时器,当前的倒计时数值作为返回值。
CLKOUT输出
???????CLKOUT引脚有可编程方波。由CLKOUT频率寄存器(地址0DH;见表23)控制操作。时钟频率32.768KHz(默认),1024,32和1Hz的频率可以作为系统时钟,单片机的时钟,输入到电荷泵,或校准振荡器。CLKOUT开漏输出,上电时启用。如果禁用它变为高阻抗。
复位低电压检测器和时钟监视器
???????PCF8563的包括内部复位电路,振荡器停止时,复位电路激活。在复位状态下,I2C总线初始化,所有寄存器和地址指针被清零,VL,TD1,TD0,TESTC和AE被设置为逻辑1.
低电压检测器和时钟监视器
???????PCF8563芯片的低电压探测器。当VDD低于Vlow,VL位(秒寄存器第7位)设置表明可靠的时钟/日历信息将不再保证.VL标志只能由软件清零。 ???????位VL用于检测在电池供电情况下,VDD慢慢降低到Vlow的情况。在VDD低于Vlow之前应该把VDD从新拉高。这种情况下,时间可能被损坏。
寄存器结构
寄存器概述
表4 位标记"-": 无效; "0"标记的,应始终与写入逻辑0.
BCD编码格式寄存器概述
表5 位标记"-": 无效; [1]不是BCD编码格式。
Control/Status 1 寄存器(内存地址00H)
表6
Control/Status 2 寄存器(内存地址01H)
表7
Seconds,Minutes,Hours寄存器
Days,Weekdays,Months/Century,Years寄存器
Alarm寄存器
???????当一个或多个报警寄存器加载一个有效的分、时、日或平日信息,其相应的AE(Alarm Enable)位为逻辑0,然后将这些信息将与当前的分、时、日和平日进行比较。当所有启用的比较第一个匹配的,位AF(Alarm Flag)置位。 ???????AF保持置位,直到通过软件清除。一旦AF被清0,它只会被重新设置时间增量,以符合再次报警条件。若报警寄存器AE置1,则该报警寄存器被忽略。
LKOUT频率寄存器
计数控制寄存器
???????8位二进制减计数器有效与否是由计数控制寄存器的TE位控制,计数器的时钟源也是由计数控制寄存器选择.计数器中断的产生与控制有状态控制寄存器2控制。为了准确读出计数器的值,I2C总线的SCL时钟至少应为计数器时钟的2倍。
EXT_CLK测试模式
???????测试模式允许进行在线测试,在测试环境下可以设置测试的条件并控制芯片的RTC. ???????通过设置控制状态寄存器1(Control/Status 1)的位TEST1进入测试环境,CLKOUT管脚转换为输入管脚。测试模式下,CLKOUT管脚的输入信号取代了芯片内部的64Hz的信号,CLKOUT管脚信号每64个上升沿产生1S增量。 ???????CLKOUT管脚提供的信号的最小脉冲带宽为300ns,最小周期为1000ns。内部64Hz时钟,或者外部信号源被内部预分频器分为1Hz,使用STOP预分频器可设置成已知状态,当位STOP置位,预分频器复位为0。在预分频器再次工作前,STOP位必须先清0.从STOP状态,第一个1S的占用CLKOUT信号32个上升沿,之后每1S的增量占用64个上升沿。 ???????备注:进入EXT_CLK测试模式内部64 Hz的时钟不同步,进入测试模式时,可以假设没有预分频器的状态。
- 进入EXT_CLK测试模式,设置Control/Status 1寄存器的7位TEST1=1;
- 设置Control/Status 1寄存器的5位STOP=1;
- 清除Control/Status 1寄存器的5位STOP=0;
- 设置时间寄存器(Seconds,Minutes,Hours,Days,Weekdays,Months/Century,Years)期望值;
- 向CLOCKOUT提供32个时钟脉冲;
- 读取时间寄存器,看第一次变化;
- 向CLOCKOUT提供64个时钟脉冲;
- 读取时间寄存器,看Second寄存器变化;
- 重复第7,8步骤,看额外的变量;
上电复位(POR)override模式
???????POR的持续时间和晶体振荡器的启动时间直接相关。由于长期启动这些类型的电路所需要的时间,一个禁用POR和加强硬板设备测试机制已建成。这种模式的设置要求,I2C总线引脚,SDA和SCL,如图5所示,在一个特定的顺序切换。所有的时间值是最低要求。 ???????一旦已进入override模式,芯片立即停止复位和正常运行,开始进入EXT_CLK测试模式,即通过I2C总线访问。位TESTC写逻辑0,停止override模式。只有位TESTC被设置为逻辑1,才能再次进入override模式。正常运行期间位TESTC设置逻辑0无效,只是为了防止进入POR override模式。
串行接口
???????PCF8563的串行接口是I2C总线。
PCF8563 I2C协议
寻址:在12C总线上传输的任何数据之前,首先处理设备应该响应。寻址总是进行传输启动程序后的第一个字节。 ???????PCF8563是一个从接收器或从发送器。因此时钟信号SCL只是一个输入信号,但数据信号SDA是双向线。 PCF8563的从地址 时钟/日历的读/写周期:在图11,12和13所示为不同的PCF8563的I2C总线配置读写周期。 WORD地址是一个4位的值,定义下一步要访问寄存器。WORD地址的高4位未使用。
驱动程序
???????来源:51单片机入门经验分享8-关于IIC通信详解及PCF8563芯片显示时间 —— dabing89 ???????stdint.h见【51单片机快速入门指南】1:基础知识和工程创建 ???????软件I2C程序见【51单片机快速入门指南】4: 软件 I2C
PCF8563.c
#include "PCF8563.h"
#include "./Soft_I2C/Soft_I2C.h"
uint8_t buffer[7];
sTime CurTime;
static uint8_t RTC_BinToBcd2(uint8_t BINValue)
{
uint8_t bcdhigh = 0;
while (BINValue >= 10)
{
bcdhigh++;
BINValue -= 10;
}
return ((uint8_t)(bcdhigh << 4) | BINValue);
}
static uint8_t RTC_Bcd2ToBin(uint8_t BCDValue)
{
uint8_t tmp = 0;
tmp = ((uint8_t)(BCDValue & (uint8_t)0xF0) >> (uint8_t)0x04) * 10;
return (tmp + (BCDValue & (uint8_t)0x0F));
}
void PCF8563_Write_Byte(uint8_t REG_ADD, uint8_t dat)
{
i2c_mem_write(PCF8563_ADDR, REG_ADD, &dat, 1);
}
uint8_t PCF8563_Read_Byte(uint8_t REG_ADD)
{
uint8_t ReData;
i2c_mem_read(PCF8563_ADDR, REG_ADD, &ReData, 1);
return ReData;
}
void PCF8563_Write_nByte(uint8_t REG_ADD, uint8_t num, uint8_t *pBuff)
{
i2c_mem_write(PCF8563_ADDR, REG_ADD, pBuff, num);
}
void PCF8563_Read_nByte(uint8_t REG_ADD, uint8_t num, uint8_t *pBuff)
{
i2c_mem_read(PCF8563_ADDR, REG_ADD, pBuff, num);
}
void SetRealTime(sTime* time)
{
time->mon &= ~PCF_Century_SetBitC;
buffer[0] = time->sec;
buffer[1] = time->min;
buffer[2] = time->hour;
buffer[3] = time->day;
buffer[4] = time->week;
buffer[5] = time->mon;
buffer[6] = time->year;
PCF8563_Write_nByte(PCF8563_Address_Seconds, 7, buffer);
}
void GetRealTime(sTime* time)
{
PCF8563_Read_nByte(PCF8563_Address_Seconds, 7, buffer);
buffer[0] &= PCF8563_Shield_Seconds;
buffer[1] &= PCF8563_Shield_Minutes;
buffer[2] &= PCF8563_Shield_Hours;
buffer[3] &= PCF8563_Shield_Days;
buffer[4] &= PCF8563_Shield_WeekDays;
buffer[5] &= PCF8563_Shield_Months_Century;
buffer[6] &= PCF8563_Shield_Years;
time->year = buffer[6];
time->mon = buffer[5];
time->week = buffer[4];
time->day = buffer[3];
time->hour = buffer[2];
time->min = buffer[1];
time->sec = buffer[0];
}
void PCF8563Init(void)
{
sTime InitTime = {0x22, 0x01, 0x10, 0x12,0x00,0x00,0x01};
SetRealTime(&InitTime);
PCF8563_Write_Byte(PCF8563_Address_Control_Status_1, 0x00);
PCF8563_Write_Byte(PCF8563_Address_Control_Status_2, 0x00);
}
PCF8563.h
#ifndef _PCF8563_H
#define _PCF8563_H
#include <stdint.h>
typedef struct {
uint16_t year;
uint8_t mon;
uint8_t day;
uint8_t hour;
uint8_t min;
uint8_t sec;
uint8_t week;
}sTime;
extern sTime CurTime;
#define PCF8563_Check_Data (uint8_t)0x55
#define PCF8563_ADDR (uint8_t)0x51
#define PCF8563_Write (uint8_t)0xa2
#define PCF8563_Read (uint8_t)0xa3
#define PCF8563_PowerResetEnable (uint8_t)0x08
#define PCF8563_PowerResetDisable (uint8_t)0x09
#define PCF_Century_SetBitC (uint8_t)0x80
#define PCF_Century_19xx (uint8_t)0x03
#define PCF_Century_20xx (uint8_t)0x04
#define PCF_Format_BIN (uint8_t)0x01
#define PCF_Format_BCD (uint8_t)0x02
#define PCF_Mode_Normal (uint8_t)0x05
#define PCF8563_Address_Control_Status_1 (uint8_t)0x00
#define PCF8563_Address_Control_Status_2 (uint8_t)0x01
#define PCF8563_Shield_Control_Status_1 (uint8_t)0xa8
#define PCF8563_Shield_Control_Status_2 (uint8_t)0x1f
#define PCF8563_Shield_Seconds (uint8_t)0x7f
#define PCF8563_Shield_Minutes (uint8_t)0x7f
#define PCF8563_Shield_Hours (uint8_t)0x3f
#define PCF8563_Shield_Days (uint8_t)0x3f
#define PCF8563_Shield_WeekDays (uint8_t)0x07
#define PCF8563_Shield_Months_Century (uint8_t)0x1f
#define PCF8563_Shield_Years (uint8_t)0xff
#define PCF8563_Address_Years (uint8_t)0x08
#define PCF8563_Address_Months (uint8_t)0x07
#define PCF8563_Address_Days (uint8_t)0x05
#define PCF8563_Address_WeekDays (uint8_t)0x06
#define PCF8563_Address_Hours (uint8_t)0x04
#define PCF8563_Address_Minutes (uint8_t)0x03
#define PCF8563_Address_Seconds (uint8_t)0x02
#define PCF_Control_Status_NormalMode (uint8_t)(~(1<<7))
#define PCF_Control_Status_EXT_CLKMode (uint8_t)(1<<7)
#define PCF_Control_ChipRuns (uint8_t)(~(1<<5))
#define PCF_Control_ChipStop (uint8_t)(1<<5)
#define PCF_Control_TestcClose (uint8_t)(~(1<<3))
#define PCF_Control_TestcOpen (uint8_t)(1<<3)
#define PCF_Accuracy_ClockNo (uint8_t)(1<<7)
#define PCF_Accuracy_ClockYes (uint8_t)(~(1<<7))
void PCF8563_Write_Byte (uint8_t REG_ADD, uint8_t dat);
uint8_t PCF8563_Read_Byte (uint8_t REG_ADD);
void PCF8563_Write_nByte (uint8_t REG_ADD, uint8_t num, uint8_t *pBuff);
void PCF8563_Read_nByte (uint8_t REG_ADD, uint8_t num, uint8_t *pBuff);
void SetRealTime(sTime *time);
void GetRealTime(sTime *time);
void PCF8563Init(void);
#endif
测试程序
main.c
#include "./Drivers/config.h"
#include "./Drivers/delay.h"
#include "./Drivers/GPIO.h"
#include "./Drivers/soft_uart.h"
#include <stdio.h>
char putchar(char Char)
{
TxSend(Char);
return Char;
}
#include "PCF8563.h"
void GPIO_config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.Mode = GPIO_PullUp;
GPIO_InitStructure.Pin = GPIO_Pin_0 | GPIO_Pin_1;
GPIO_Inilize(GPIO_P1,&GPIO_InitStructure);
}
void main(void)
{
GPIO_config();
PCF8563Init();
while(1)
{
GetRealTime(&CurTime);
printf("%x %x %x %x %x %x %x\r\n", (int)CurTime.year, (int)CurTime.mon, (int)CurTime.day, (int)CurTime.hour, (int)CurTime.min, (int)CurTime.sec, (int)CurTime.week);
delay_ms(500);
}
}
实验现象
???????打开串口调试助手,可见能成功读取时钟值。
|