STM32学习笔记之RTC实时时钟2

电子说

1.2w人已加入

描述

5. 程序实现

5.1 初始化

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) //检查指定的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(1972,1,2,1,1,1);  //设置时间    
        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

}
static void RTC_NVICConfig(void)
{    
    NVIC_InitTypeDef NVIC_InitStructure;
    NVIC_InitStructure.NVIC_IRQChannel = RTC_IRQn;      //RTC全局中断
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;   //先占优先级1位,从优先级3位
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;  //先占优先级0位,从优先级4位
    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;     //使能该通道中断
    NVIC_Init(&NVIC_InitStructure);     //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器
}
中断服务函数:

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_ClearITPendingBit(RTC_IT_SEC|RTC_IT_OW);        //清闹钟中断
    RTC_WaitForLastTask();                                           
}

5.2 RTC部分实现

闰年判断:

//判断是否是闰年函数
//月份   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;t
得到当前时间:

//得到当前的时间
//返回值:0,成功;其他:错误代码.
u8 RTC_Get(void)
{
    static u16 daycnt=0;
    u32 timecount=0; 
    u32 temp=0;
    u16 temp1=0;

    timecount = RTC_GetCounter();     
    //timecount = 86400*4+88;
     temp = timecount / 86400;   //得到天数(秒钟数对应的)
    if(daycnt!=temp)//超过一天了
    {     
        daycnt = temp;
        temp1 = 1970;   //从1970年开始
        while(temp>=365)
        {                
            if(Is_Leap_Year(temp1))//是闰年
            {
                if(temp>=366)
                    temp -= 366;//减去一闰年,还剩下的天数
                else 
                    {   
                        temp1 ++;
                        break;
                    }  
            }
            else 
                temp -= 365;      //减去一平年 ,还剩下的天数
            temp1 ++;  
        }   
        calendar.w_year = temp1;//得到年份
        temp1=0;
        while(temp>=28)//28天,超过了最小的一个月
        {
            if(Is_Leap_Year(calendar.w_year)&&temp1==1)//当年是不是闰年/2月份
            {
                if(temp>=29)
                    temp -= 29;//闰年的秒钟数
                else 
                    break; 
            }
            else 
            {
                if(temp>=mon_table[temp1])
                    temp -= mon_table[temp1];//平年
                else 
                    break;
            }
            temp1++;  
        }
        calendar.w_month = temp1+1; //得到月份
        calendar.w_date = temp+1;   //得到日期 
    }
    temp = timecount%86400;             //得到秒钟数        
    calendar.hour = temp/3600;      //小时
    calendar.min = (temp%3600)/60;  //分钟    
    calendar.sec = (temp%3600)%60;  //秒钟
    calendar.week = RTC_Get_Week(calendar.w_year,calendar.w_month,calendar.w_date);//获取星期 

    return 0;
}
得到星期几:

//获得现在是星期几
//功能描述:输入公历日期得到星期(只允许1901-2099年)
//输入参数:公历年月日 
//返回值:星期号                                                                                         
u8 RTC_GetWeek(u16 year,u8 month,u8 day)
{    
    u16 temp2;
    u8 yearH,yearL;

    yearH = year/100;
    yearL = year%100; 
    // 如果为21世纪,年份数加100  
    if (yearH>19)
        yearL += 100;
    // 所过闰年数只算1900年之后的  
    temp2 = yearL+yearL/4;
    temp2 = temp2%7; 
    temp2 = temp2+day+table_week[month-1];
    if (yearL%4==0&&month<3)
        temp2--;

    return(temp2%7);
}

6. 附录:

STM32 时钟树:

计数器

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分