超透彻的STM32讲解资料RTC时钟.docx

上传人:b****3 文档编号:2612185 上传时间:2022-11-03 格式:DOCX 页数:14 大小:26.03KB
下载 相关 举报
超透彻的STM32讲解资料RTC时钟.docx_第1页
第1页 / 共14页
超透彻的STM32讲解资料RTC时钟.docx_第2页
第2页 / 共14页
超透彻的STM32讲解资料RTC时钟.docx_第3页
第3页 / 共14页
超透彻的STM32讲解资料RTC时钟.docx_第4页
第4页 / 共14页
超透彻的STM32讲解资料RTC时钟.docx_第5页
第5页 / 共14页
点击查看更多>>
下载资源
资源描述

超透彻的STM32讲解资料RTC时钟.docx

《超透彻的STM32讲解资料RTC时钟.docx》由会员分享,可在线阅读,更多相关《超透彻的STM32讲解资料RTC时钟.docx(14页珍藏版)》请在冰豆网上搜索。

超透彻的STM32讲解资料RTC时钟.docx

这部分的内容实现的功能是将时间传输到上位机。

第一:

串口的配置,前面已经详细的讲过,这里不再讲解。

第二:

中断的配置,由于我们需要时间每秒自加一次,一次需要用到中断,在中断服务程序中实现秒的自加,前面也已讲过中断的配置,这里就不详细讲解,只给出代码

/*

*函数名:

NVIC_Configuration

*描述:

配置RTC秒中断的主中断优先级为1,次优先级为0

*输入:

*输出:

*调用:

外部调用

*/

voidNVIC_Configuration(void)

{

NVIC_InitTypeDefNVIC_InitStructure;

/*设置先占优先级1位,从占优先级3位*/

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);

/*选择RTC的IRQ通道*/

NVIC_InitStructure.NVIC_IRQChannel=RTC_IRQn;

/*设置中断先占优先级为1*/

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1;

/*设置中断从占优先级为1*/

NVIC_InitStructure.NVIC_IRQChannelSubPriority=0;

/*使能RTC的IRQ通道*/

NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE;

NVIC_Init(&NVIC_InitStructure);

}

第三:

对于RTC的使用,首先我们要判断是否是第一次使用RTC,如果是第一次使用那么肯定要设置时间的初始值以及对RTC进行相应的配置,而如果不是第一次使用,那么我们就无需再设置时间的初始值以及对RTC进行相应的配置,只需让RTC计数器继续计数就可以了。

那么我们如何才能判断RTC是否为第一次使用呢?

STM32中有一个后备寄存器,寄存器中的值不会因为掉电而改变,既然如此那我们肯定会这样想,当我第一次使用RTC时,往后备寄存器中写入一个值,下次再使用RTC时,我只要判断后备寄存器中的值是否为我第一次用RTC时写入的值,如果相等,说明我以前已经用过RTC了,现在我无需再对RTC进行配置了,因为第一次都配置好了(RTC和后备寄存器一样,RTC寄存器中设置的值不会因为掉电而改变),但要注意的是RTC的允许中断这一位在每次复位后会回到默认值,所以每次复位后我们都要再次设置允许RTC中断。

下面我们就看看RTC的配置程序:

/*

*函数名:

RTC_Configuration

*描述:

配置RTC

*输入:

*输出:

*调用:

外部调用

*/

voidRTC_Configuration(void)

{

/*使能PWR和BKP时钟*/

RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|RCC_APB1Periph_BKP,ENABLE);

/*取消后备区域的写保护,因为后备寄存器中放的是重要的数据,默认是不允许往里面写入值的*/

PWR_BackupAccessCmd(ENABLE);

/*将后背寄存器的寄存器值设为默认值*/

BKP_DeInit();

/*打开外部低速晶振,RTC可以选择的时钟源是外部和内部低速晶振及外部高速晶振,这里我们选择外部低速晶振32768HZ*/

RCC_LSEConfig(RCC_LSE_ON);

/*等待外部低速晶振准备就序*/

while(RCC_GetFlagStatus(RCC_FLAG_LSERDY)==RESET)

{}

/*选择外部低速晶振为RTC的时钟源*/

RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);

/*EnableRTCClock*/

RCC_RTCCLKCmd(ENABLE);

/*等待RTC寄存器与RTC的APB时钟同步*/

RTC_WaitForSynchro();

/*等待上次对RTC寄存器配置完成*/

RTC_WaitForLastTask();

/*使能RTC中断*/

RTC_ITConfig(RTC_IT_SEC,ENABLE);

/*等待上次对RTC寄存器配置完成*/

RTC_WaitForLastTask();

/*设置RTC的预分频值,因为外部低速晶振是32768,所以选择*/

/*RTC计数器计数频率=RTCCLK/RTC_PR=(32.768KHz)/(32767+1)*/

RTC_SetPrescaler(32767);

/*等待上次对RTC寄存器配置完成*/

RTC_WaitForLastTask();

}

下面再来看看在RTC中断中写了哪些程序:

/*

*函数名:

RTC_IRQHandler(void)

*描述:

RTC中断服务函数

*输入:

*输出:

*调用:

*/

voidRTC_IRQHandler(void)

{/*判断中断标志位是否被置位*/

if(RTC_GetITStatus(RTC_IT_SEC)!

=RESET)

{

/*清除中断标志位*/

RTC_ClearITPendingBit(RTC_IT_SEC);

/*TimeDisplay是一个标志位,只有等于1时才让串口发送时间数据,即让串口一秒发一次时间值*/

TimeDisplay=1;

/*WaituntillastwriteoperationonRTCregistershasfinished*/

RTC_WaitForLastTask();

/*当时间走到23:

59:

59秒时RTC计数器中的值清零,0x00015180=23*3600+56*60+59*/

if(RTC_GetCounter()==0x00015180)

{

RTC_SetCounter(0x0);

/*WaituntillastwriteoperationonRTCregistershasfinished*/

RTC_WaitForLastTask();

}

}

}

第四:

RTC配置好后肯定要对初始时间的设置进行相应的配置了,首先是如何从串口中接受设置的时间初始值:

/*

*函数名:

USART_Scanf

*描述:

串口从超级终端中获取数值

*输入:

-value用户在超级终端中输入的数值

*输出:

*调用:

被Time_Regulate(void)调用

*/

uint8_tUSART_Scanf(uint32_tvalue)

{

uint32_tindex=0;

uint32_ttmp[2]={0,0};

while(index<2)

{

/*等待数据接受完成*/

while(USART_GetFlagStatus(USART1,USART_FLAG_RXNE)==RESET)

{}

/*将数据给数组tmp,从串口终端里面输进去的数是ASCII码值

*/

tmp[index++]=(USART_ReceiveData(USART1));

/*判断接受到的数据是否在0到9之间*/

if((tmp[index-1]<0x30)||(tmp[index-1]>0x39))

{

printf("\n\rPleaseentervalidnumberbetween0and9");

index--;

}

}

/*将接受的两个数据组成一个两位数*/

index=(tmp[1]-0x30)+((tmp[0]-0x30)*10);

/*判断组成后的值是否有无,比如组成的小时不能超过23,分秒不能超过59,value即为23,或59*/

if(index>value)

{

printf("\n\rPleaseentervalidnumberbetween0and%d",value);

return0xFF;

}

returnindex;

}

上面的函数是被下面的Time_Regulate(void)函数调用的。

/*

*函数名:

Time_Regulate

*描述:

返回用户在超级终端中输入的时间值,并将值储存在

*RTC计数寄存器中。

*输入:

*输出:

用户在超级终端中输入的时间值,单位为s

*调用:

被Time_Adjust调用,调用USART_Scanf

*/

uint32_tTime_Regulate(void)

{

uint32_tTmp_HH=0xFF,Tmp_MM=0xFF,Tmp_SS=0xFF;

printf("\r\n==============TimeSettings=====================================");

printf("\r\nPleaseSetHours");

while(Tmp_HH==0xFF)

{

/*将串口接受的数据给Tmp_HH,23便是传递给USART_Scanf(value)中的形参value的*/

Tmp_HH=USART_Scanf(23);

}

printf(":

%d",Tmp_HH);

printf("\r\nPleaseSetMinutes");

while(Tmp_MM==0xFF)

{

Tmp_MM=USART_Scanf(59);

}

printf(":

%d",Tmp_MM);

printf("\r\nPleaseSetSeconds");

while(Tmp_SS==0xFF)

{

Tmp_SS=USART_Scanf(59);

}

printf(":

%d",Tmp_SS);

/*将时分秒转换成秒放入RTC计数器中*/

return((Tmp_HH*3600+Tmp_MM*60+Tmp_SS));

}

/*

*函数名:

Time_Adjust

*描述:

时间调节

*输入:

*输出:

*调用:

主函数调用

*/

voidTime_Adjust(void)

{

/*WaituntillastwriteoperationonRTCregistershasfinished*/

RTC_WaitForLastTask();

/*将设置的初始时间值装入RTC计数器,RTC开始运行时,计数器里面的值会在初始值的基础上自动一秒加1次*/

RTC_SetCounter(Time_Regulate());

/*WaituntillastwriteoperationonRTCregistershasfinished*/

RTC_WaitForLast

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

当前位置:首页 > 高等教育 > 历史学

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

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