当前位置:首页 > 单片机 > 单片机
[导读]RCT特征:可编程的预分频系数,分频系数最高2的20次方。32位可编程计数,用于较长时间段的测量。2个分离的时钟。可以选择三种RTC时钟源:HSE/128;LSE振荡器;LSI振荡器。2个独立的复位类型:APB1由系统复位;RTC由后

RCT特征:

可编程的预分频系数,分频系数最高2的20次方。

32位可编程计数,用于较长时间段的测量。

2个分离的时钟。

可以选择三种RTC时钟源:HSE/128;LSE振荡器;LSI振荡器。

2个独立的复位类型:APB1由系统复位;RTC由后备域复位。

三个专门的可屏蔽中断:闹钟中断;秒中断(一个可编程周期,最长可达1s);溢出中断。

RTC工作原理框图

RTCCLK经过RTC_DIV预分频,RTC_PRL设置预分频系数,然后得到TR_CLK时钟信号,我们一般设置其周期为1s,RTC_CNT计数器计数,假如1970设置为时间起点为0s,通过当前时间的秒数计算得到当前的时间。RTC_ALR是设置闹钟时间,RTC_CNT计数到RTC_ALR就会产生计数中断,RTC_Second为秒中断,用于刷新时间,RTC_Overflow是溢出中断。

实验例程:
#include"sys.h"#include"delay.h"#include"usart.h"#include"rtc.h"_calendar_objcalendar;//时钟结构体staticvoidRTC_NVIC_Config(void){NVIC_InitTypeDefNVIC_InitStructure;NVIC_InitStructure.NVIC_IRQChannel=RTC_IRQn;//RTC全局中断NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=0;NVIC_InitStructure.NVIC_IRQChannelSubPriority=0;NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;//使能该通道中断NVIC_Init(&NVIC_InitStructure);//根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器}//实时时钟配置//初始化RTC时钟,同时检测时钟是否工作正常//BKP->DR1用于保存是否第一次配置的设置//返回0:正常//其他:错误代码u8 RTC_Init(void){    //检查是不是第一次配置时钟    u8 temp=0;    RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE);    //使能PWR和BKP外设时钟       PWR_BackupAccessCmd(ENABLE);    //使能后备寄存器访问      if (BKP_ReadBackupRegister(BKP_DR1) != 0x5050)      //从指定的后备寄存器中读出数据:读出了与写入的指定数据不相乎        {                       BKP_DeInit();   //复位备份区域            RCC_LSEConfig(RCC_LSE_ON);  //设置外部低速晶振(LSE),使用外设低速晶振        while (RCC_GetFlagStatus(RCC_FLAG_LSERDY)==RESET&&temp<250) //检查指定的RCC标志位设置与否,等待低速晶振就绪            {            temp++;            delay_ms(10);            }        if(temp>=250)return 1;//初始化时钟失败,晶振有问题               RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);     //设置RTC时钟(RTCCLK),选择LSE作为RTC时钟            RCC_RTCCLKCmd(ENABLE);  //使能RTC时钟          RTC_WaitForLastTask();  //等待最近一次对RTC寄存器的写操作完成        RTC_WaitForSynchro();       //等待RTC寄存器同步          RTC_ITConfig(RTC_IT_SEC, ENABLE);       //使能RTC秒中断        RTC_WaitForLastTask();  //等待最近一次对RTC寄存器的写操作完成        RTC_EnterConfigMode();/// 允许配置          RTC_SetPrescaler(32767); //设置RTC预分频的值        RTC_WaitForLastTask();  //等待最近一次对RTC寄存器的写操作完成        RTC_Set(2015,1,14,17,42,55);  //设置时间            RTC_ExitConfigMode(); //退出配置模式          BKP_WriteBackupRegister(BKP_DR1, 0X5050);   //向指定的后备寄存器中写入用户程序数据        }    else//系统继续计时        {        RTC_WaitForSynchro();   //等待最近一次对RTC寄存器的写操作完成        RTC_ITConfig(RTC_IT_SEC, ENABLE);   //使能RTC秒中断        RTC_WaitForLastTask();  //等待最近一次对RTC寄存器的写操作完成        }    RTC_NVIC_Config();//RCT中断分组设置                                    RTC_Get();//更新时间        return 0; //ok}                           //RTC时钟中断//每秒触发一次  //extern u16 tcnt; void RTC_IRQHandler(void){            if (RTC_GetITStatus(RTC_IT_SEC) != RESET)//秒钟中断    {                                   RTC_Get();//更新时间       }    if(RTC_GetITStatus(RTC_IT_ALR)!= RESET)//闹钟中断    {        RTC_ClearITPendingBit(RTC_IT_ALR);      //清闹钟中断           RTC_Get();                //更新时间       printf("Alarm Time:%d-%d-%d %d:%d:%dn",calendar.w_year,calendar.w_month,calendar.w_date,calendar.hour,calendar.min,calendar.sec);//输出闹铃时间      }                                                    RTC_ClearITPendingBit(RTC_IT_SEC|RTC_IT_OW);        //清闹钟中断    RTC_WaitForLastTask();                                           }//判断是否是闰年函数//月份   1  2  3  4  5  6  7  8  9  10 11 12//闰年   31 29 31 30 31 30 31 31 30 31 30 31//非闰年 31 28 31 30 31 30 31 31 30 31 30 31//输入:年份//输出:该年份是不是闰年.1,是.0,不是u8 Is_Leap_Year(u16 year){                 if(year%4==0) //必须能被4整除    {         if(year%100==0)         {             if(year%400==0)return 1;//如果以00结尾,还要能被400整除                    else return 0;           }else return 1;       }else return 0; }                  //设置时钟//把输入的时钟转换为秒钟//以1970年1月1日为基准//1970~2099年为合法年份//返回值:0,成功;其他:错误代码.//月份数据表                                          u8 const table_week[12]={0,3,3,6,1,4,6,2,5,0,3,5}; //月修正数据表   //平年的月份日期表const u8 mon_table[12]={31,28,31,30,31,30,31,31,30,31,30,31};u8 RTC_Set(u16 syear,u8 smon,u8 sday,u8 hour,u8 min,u8 sec){    u16 t;    u32 seccount=0;    if(syear<1970||syear>2099)return 1;        for(t=1970;t2099)return 1;        for(t=1970;t		
本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭