基于STM32 RTC实时时钟.docx

上传人:b****3 文档编号:2133223 上传时间:2022-10-27 格式:DOCX 页数:16 大小:149.89KB
下载 相关 举报
基于STM32 RTC实时时钟.docx_第1页
第1页 / 共16页
基于STM32 RTC实时时钟.docx_第2页
第2页 / 共16页
基于STM32 RTC实时时钟.docx_第3页
第3页 / 共16页
基于STM32 RTC实时时钟.docx_第4页
第4页 / 共16页
基于STM32 RTC实时时钟.docx_第5页
第5页 / 共16页
点击查看更多>>
下载资源
资源描述

基于STM32 RTC实时时钟.docx

《基于STM32 RTC实时时钟.docx》由会员分享,可在线阅读,更多相关《基于STM32 RTC实时时钟.docx(16页珍藏版)》请在冰豆网上搜索。

基于STM32 RTC实时时钟.docx

基于STM32RTC实时时钟

1课程设计内容

本文将利用ALIENTEK2.8寸TFTLCD模块来显示日期时间,实现一个简单的时钟。

2STM32芯片简介

2006年ARM公司推出了基于ARMv7架构的Cortex系列的标准体系结构,以满足各种技术的不同性能要求,包含A、R、M三个分工明确的系列[1]。

其中,A系列面向复杂的尖端应用程序,用于运行开放式的复杂操作系统;R系列适合实时系统;M系列则专门针对低成本的微控制领域。

Cortex-M3是首款基于ARMv7-M体系结构的32位标准处理器,具有低功耗、少门数、短中断延迟、低调试成本等众多优点。

它是专门为在微控制系统、汽车车身系统、工业控制系统和无线网络等对功耗和成本敏感的嵌入式应用领域实现高系统性能而设计的,它大大简化了编程的复杂性,集高性能、低功耗、低成本于一体[2]。

半导体制造厂商意法半导体ST公司是ARM公司Cortex-M3内核开发项目一个主要合作方,2007年6月11日ST公司率先推出了基于Cortex-M3内核的STM32系列MCU。

本章将简要介绍STM32系列处理器的分类、内部结构及特点,并对本设计中重点应用的通用定时器做进一步分析。

2.1STM32RTC时钟简介

STM32的实时时钟(RTC)是一个独立的定时器。

STM32的RTC模块拥有一组连续计数的计数器,在相应软件配置下,可提供时钟日历的功能。

修改计数器的值可以重新设置系统当前的时间和日期。

RTC模块和时钟配置系统(RCC_BDCR寄存器)是在后备区域,即在系统复位或从待机模式唤醒后RTC的设置和时间维持不变。

但是在系统复位后,会自动禁止访问后备寄存器和RTC,以防止对后备区域(BKP)的意外写操作。

所以在要设置时间之前,先要取消备份区域(BKP)写保护。

RTC的简化框图,如图20.1.1所示:

图20.1.1RTC框图

RTC由两个主要部分组成(参见图20.1.1),第一部分(APB1接口)用来和APB1总线相连。

此单元还包含一组16位寄存器,可通过APB1总线对其进行读写操作。

APB1接口由APB1总线时钟驱动,用来与APB1总线连接。

另一部分(RTC核心)由一组可编程计数器组成,分成两个主要模块。

第一个模块是RTC的预分频模块,它可编程产生1秒的RTC时间基准TR_CLK。

RTC的预分频模块包含了一个20位的可编程分频器(RTC预分频器)。

如果在RTC_CR寄存器中设置了相应的允许位,则在每个TR_CLK周期中RTC产生一个中断(秒中断)。

第二个模块是一个32位的可编程计数器,可被初始化为当前的系统时间,一个32位的时钟计数器,按秒钟计算,可以记录4294967296秒,约合136年左右,作为一般应用,这已经是足够了的。

RTC还有一个闹钟寄存器RTC_ALR,用于产生闹钟。

系统时间按TR_CLK周期累加并与存储在RTC_ALR寄存器中的可编程时间相比较,如果RTC_CR控制寄存器中设置了相应允许位,比较匹配时将产生一个闹钟中断。

RTC内核完全独立于RTCAPB1接口,而软件是通过APB1接口访问RTC的预分频值、计数器值和闹钟值的。

但是相关可读寄存器只在RTCAPB1时钟进行重新同步的RTC时钟的上升沿被更新,RTC标志也是如此。

这就意味着,如果APB1接口刚刚被开启之后,在第一次的内部寄存器更新之前,从APB1上都处的RTC寄存器值可能被破坏了(通常读到0)。

因此,若在读取RTC寄存器曾经被禁止的RTCAPB1接口,软件首先必须等待RTC_CRL寄存器的RSF位(寄存器同步标志位,bit3)被硬件置1。

2.2RTC相关配置

正常工作的一般配置步骤如下:

1)使能电源时钟和备份区域时钟。

前面已经介绍了,我们要访问RTC和备份区域就必须先使能电源时钟和备份区域时钟。

RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|RCC_APB1Periph_BKP,ENABLE);

2)取消备份区写保护。

要向备份区域写入数据,就要先取消备份区域写保护(写保护在每次硬复位之后被使能),否则是无法向备份区域写入数据的。

我们需要用到向备份区域写入一个字节,来标记时钟已经配置过了,这样避免每次复位之后重新配置时钟。

取消备份区域写保护的库函数实现方法是:

PWR_BackupAccessCmd(ENABLE);//使能RTC和后备寄存器访问

3)复位备份区域,开启外部低速振荡器。

在取消备份区域写保护之后,我们可以先对这个区域复位,以清除前面的设置,当然这个操作不要每次都执行,因为备份区域的复位将导致之前存在的数据丢失,所以要不要复位,要看情况而定。

然后我们使能外部低速振荡器,注意这里一般要先判断RCC_BDCR的LSERDY位来确定低速振荡器已经就绪了才开始下面的操作。

备份区域复位的函数是:

BKP_DeInit();//复位备份区域

开启外部低速振荡器的函数是:

RCC_LSEConfig(RCC_LSE_ON);//开启外部低速振荡器

4)选择RTC时钟,并使能。

这里我们将通过RCC_BDCR的RTCSEL来选择选择外部LSI作为RTC的时钟。

然后通过RTCEN位使能RTC时钟。

库函数中,选择RTC时钟的函数是:

RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);//选择LSE作为RTC时钟

对于RTC时钟的选择,还有RCC_RTCCLKSource_LSI和RCC_RTCCLKSource_HSE_Div128

两个,顾名思义,前者为LSI,后者为HSE的128分频,这在时钟系统章节有讲解过。

使能RTC时钟的函数是:

RCC_RTCCLKCmd(ENABLE);//使能RTC时钟

5)设置RTC的分频,以及配置RTC时钟。

在开启了RTC时钟之后,我们要做的就是设置RTC时钟的分频数,通过RTC_PRLH和RTC_PRLL来设置,然后等待RTC寄存器操作完成,并同步之后,设置秒钟中断。

然后设置RTC的允许配置位(RTC_CRH的CNF位),设置时间(其实就是设置RTC_CNTH和RTC_CNTL两个寄存器)。

下面我们一一这些步骤用到的库函数:

在进行RTC配置之前首先要打开允许配置位(CNF),库函数是:

RTC_EnterConfigMode();///允许配置

在配置完成之后,千万别忘记更新配置同时退出配置模式,函数是:

RTC_ExitConfigMode();//退出配置模式,更新配置设置RTC时钟分频数,库函数是:

voidRTC_SetPrescaler(uint32_tPrescalerValue);

这个函数只有一个入口参数,就是RTC时钟的分频数,很好理解。

然后是设置秒中断允许,RTC使能中断的函数是:

voidRTC_ITConfig(uint16_tRTC_IT,FunctionalStateNewState);

这个函数的第一个参数是设置秒中断类型,这些通过宏定义定义的。

对于使能秒中断方法是:

RTC_ITConfig(RTC_IT_SEC,ENABLE);//使能RTC秒中断

下一步便是设置时间了,设置时间实际上就是设置RTC的计数值,时间与计数值之间是需要换算的。

库函数中设置RTC计数值的方法是:

voidRTC_SetCounter(uint32_tCounterValue)最后在配置完成之后

通过这个函数直接设置RTC计数值。

6)更新配置,设置RTC中断分组。

在设置完时钟之后,我们将配置更新同时退出配置模式,这里还是通过RTC_CRH的CNF来实现。

库函数的方法是:

RTC_ExitConfigMode();//退出配置模式,更新配置

在退出配置模式更新配置之后我们在备份区域BKP_DR1中写入0X5050代表我们已经初始化过时钟了,下次开机(或复位)的时候,先读取BKP_DR1的值,然后判断是否是0X5050来决定是不是要配置。

接着我们配置RTC的秒钟中断,并进行分组。

往备份区域写用户数据的函数是:

voidBKP_WriteBackupRegister(uint16_tBKP_DR,uint16_tData);

这个函数的第一个参数就是寄存器的标号了,这个是通过宏定义定义的。

比如我们要往BKP_DR1写入0x5050,方法是:

BKP_WriteBackupRegister(BKP_DR1,0X5050);

同时,有写便有读,读取备份区域指定寄存器的用户数据的函数是:

uint16_tBKP_ReadBackupRegister(uint16_tBKP_DR);

这个函数就很好理解了,这里不做过多讲解。

设置中断分组的方法之前已经详细讲解过,调用NVIC_Init函数即可,这里不做重复讲解。

7)编写中断服务函数。

最后,我们要编写中断服务函数,在秒钟中断产生的时候,读取当前的时间值,并显示到TFTLCD模块上。

通过以上几个步骤,我们就完成了对RTC的配置,并通过秒钟中断来更新时间。

3单元模块及电路设计

3.1电源模块

图1

3.2复位电路模块

图2

3.3外部时钟模块

图3

3.4外部晶振模块

图4

3.5JTAG下载模块

图5

3.6主控制器模块

图8

3.7BootLoader配置模块

图9

4软件设计

首先是RTC_Init,其代码如下:

//实时时钟配置

//初始化RTC时钟,同时检测时钟是否工作正常

//BKP->DR1用于保存是否第一次配置的设置

//返回0:

正常

//其他:

错误代码

u8RTC_Init(void)

{

u8temp=0;//检查是不是第一次配置时钟

if(BKP_ReadBackupRegister(BKP_DR1)!

=0x5050)//从指定的后备寄存器中

//读出数据:

读出了与写入的指定数据不相乎

{

RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|

RCC_APB1Periph_BKP,ENABLE);//使能PWR和BKP外设时钟

PWR_BackupAccessCmd(ENABLE);//使能后备寄存器访问

BKP_DeInit();//③复位备份区域

RCC_LSEConfig(RCC_LSE_ON);//设置外部低速晶振(LSE)

while(RCC_GetFlagStatus(RCC_FLAG_LSERDY)==RESET)//检查指定的

//RCC标志位设置与否,等待低速晶振就绪

{

temp++;

delay_ms(10);

}

if(temp>=250)return1;//初始化时钟失败,晶振有问题

RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);//设置RTC时钟

//(RTCCLK),选择LSE作为RTC时钟

RCC_RTCCLKCmd(ENABLE);//使能RTC时钟

RTC_WaitForLastTask();//等待最近一次对RTC寄存器的写操作完成

RTC_WaitForSynchro();//等待RTC寄存器同步

RTC_ITC

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 表格模板 > 合同协议

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1