OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 2806|回复: 6

RTC时钟软件仿真能正常工作,硬件仿真却不行,是怎么回事?

[复制链接]

8

主题

43

帖子

0

精华

初级会员

Rank: 2

积分
102
金钱
102
注册时间
2015-8-19
在线时间
3 小时
发表于 2015-9-27 09:32:40 | 显示全部楼层 |阅读模式
5金钱
求助,RTC时钟软件能够仿真,但到了硬件就运行不了
int main(void)
 {
    u32 t=10;
  char str_year[25],str_month[25],str_date[25],str_hour[25],str_min[25],str_sec[25];
//  USART1_Config();
// RTC_NVIC_Config();
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);

LCD_Init();
RTC_Init();  

LCD_Clear(WHITE);
    sprintf(str_month, "%d" ,calendar.w_month);

 while(1) 
  {
if(t!=calendar.sec)
 {
   t=calendar.sec;
   sprintf(str_year, "%d" ,10);  
//      sprintf(str_month, "%d" ,calendar.w_month);
        sprintf(str_date, "%d" ,calendar.w_date);
        sprintf(str_hour, "%d" ,calendar.hour);
        sprintf(str_min, "%d" ,calendar.min);
        sprintf(str_sec, "%d" ,calendar.sec);
LCD_ShowString(30,50, str_year,POINT_COLOR);  
LCD_ShowString(70,50,str_month,POINT_COLOR);  
LCD_ShowString(86,50,str_date,POINT_COLOR);  
// LCD_ShowString(30,50,&calendar.w_date,POINT_COLOR);  
switch(calendar.week)
{
case 0:
LCD_ShowString(30,70,"Sunday   ",POINT_COLOR);
break;
case 1:
LCD_ShowString(30,70,"Monday   ",POINT_COLOR);
break;
case 2:
LCD_ShowString(30,70,"Tuesday  ",POINT_COLOR);
break;
case 3:
LCD_ShowString(30,70,"Wednesday",POINT_COLOR);
break;
case 4:
LCD_ShowString(30,70,"Thursday ",POINT_COLOR);
break;
case 5:
LCD_ShowString(30,70,"Friday   ",POINT_COLOR);
break;
case 6:
LCD_ShowString(30,70,"Saturday ",POINT_COLOR);
break;  
}
LCD_ShowString(30,90,str_hour,POINT_COLOR);  
LCD_ShowString(54,90,str_min,POINT_COLOR);  
LCD_ShowString(78,90,str_sec,POINT_COLOR);
 }
delay_ms(10);
  }
}

最佳答案

查看完整内容[请看2#楼]

找到原因了。delay的太短 delay_ms(10);-》delay_ms(100) 就可以了,还是得怪自己粗心大意
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

8

主题

43

帖子

0

精华

初级会员

Rank: 2

积分
102
金钱
102
注册时间
2015-8-19
在线时间
3 小时
 楼主| 发表于 2015-9-27 09:32:41 | 显示全部楼层
找到原因了。delay的太短
delay_ms(10);-》delay_ms(100)
就可以了,还是得怪自己粗心大意
回复

使用道具 举报

8

主题

43

帖子

0

精华

初级会员

Rank: 2

积分
102
金钱
102
注册时间
2015-8-19
在线时间
3 小时
 楼主| 发表于 2015-9-27 09:33:56 | 显示全部楼层
RTC的程序完全是例子中的程序,没有任何的修改,
//#include "sys.h"
//#include "delay.h"
#include "usart1.h"
#include "rtc.h" 
#include "text.h"    
//Mini STM32开发板
//RTC实时时钟 驱动代码  
//正点原子@ALIENTEK
//2010/6/6
  
_calendar_obj calendar;//时钟结构体 
 
static void RTC_NVIC_Config(void) //static
{
  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寄存器
}

//实时时钟配置
//初始化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:%d\n",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;t<syear;t++) //把所有年份的秒钟相加
{
if(Is_Leap_Year(t))seccount+=31622400;//闰年的秒钟数
else seccount+=31536000;  //平年的秒钟数
}
smon-=1;
for(t=0;t<smon;t++)   //把前面月份的秒钟数相加
{
seccount+=(u32)mon_table[t]*86400;//月份秒钟数相加
if(Is_Leap_Year(syear)&&t==1)seccount+=86400;//闰年2月份增加一天的秒钟数   
}
seccount+=(u32)(sday-1)*86400;//把前面日期的秒钟数相加 
seccount+=(u32)hour*3600;//小时秒钟数
    seccount+=(u32)min*60; //分钟秒钟数
seccount+=sec;//最后的秒钟加上去

RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE); //使能PWR和BKP外设时钟  
PWR_BackupAccessCmd(ENABLE); //使能RTC和后备寄存器访问 
RTC_SetCounter(seccount); //设置RTC计数器的值

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

//初始化闹钟  
//以1970年1月1日为基准
//1970~2099年为合法年份
//syear,smon,sday,hour,min,sec:闹钟的年月日时分秒   
//返回值:0,成功;其他:错误代码.
u8 RTC_Alarm_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<syear;t++) //把所有年份的秒钟相加
{
if(Is_Leap_Year(t))seccount+=31622400;//闰年的秒钟数
else seccount+=31536000;  //平年的秒钟数
}
smon-=1;
for(t=0;t<smon;t++)   //把前面月份的秒钟数相加
{
seccount+=(u32)mon_table[t]*86400;//月份秒钟数相加
if(Is_Leap_Year(syear)&&t==1)seccount+=86400;//闰年2月份增加一天的秒钟数   
}
seccount+=(u32)(sday-1)*86400;//把前面日期的秒钟数相加 
seccount+=(u32)hour*3600;//小时秒钟数
    seccount+=(u32)min*60; //分钟秒钟数
seccount+=sec;//最后的秒钟加上去    
//设置时钟
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE); //使能PWR和BKP外设时钟   
PWR_BackupAccessCmd(ENABLE); //使能后备寄存器访问  
//上面三步是必须的!

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

return 0;    
}


//得到当前的时间
//返回值:0,成功;其他:错误代码.
u8 RTC_Get(void)
{
static u16 daycnt=0;
u32 timecount=0; 
u32 temp=0;
u16 temp1=0;  
    timecount=RTC_GetCounter();  
  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)//超过了一个月
{
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_Get_Week(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);
}  



回复

使用道具 举报

8

主题

43

帖子

0

精华

初级会员

Rank: 2

积分
102
金钱
102
注册时间
2015-8-19
在线时间
3 小时
 楼主| 发表于 2015-9-27 09:36:56 | 显示全部楼层
我把主函数中的t改为0之后就什么都不显示,说明没有运行到if里面,就是calendar.sec=0,为什么会等于0?实在搞不懂,求助各位大神
回复

使用道具 举报

22

主题

180

帖子

1

精华

高级会员

Rank: 4

积分
616
金钱
616
注册时间
2015-6-29
在线时间
101 小时
发表于 2015-9-27 10:24:42 | 显示全部楼层
如果代码完全一模一样,难道是RTC不起振?
我是菜鸟
回复

使用道具 举报

8

主题

43

帖子

0

精华

初级会员

Rank: 2

积分
102
金钱
102
注册时间
2015-8-19
在线时间
3 小时
 楼主| 发表于 2015-9-27 11:06:54 | 显示全部楼层
回复【4楼】负西弱:
---------------------------------
硬件没问题
回复

使用道具 举报

8

主题

43

帖子

0

精华

初级会员

Rank: 2

积分
102
金钱
102
注册时间
2015-8-19
在线时间
3 小时
 楼主| 发表于 2015-9-27 11:46:46 | 显示全部楼层
实在找不到原因,求帮助
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-18 18:25

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表