当前位置: 首页 > news >正文

STM32+FATFS+SD卡+RTC(生成.CSV格式文件)

一、简介

实验目的:在SD卡上挂载文件系统,实时记录压力传感器采集到的数据;且在表格第一排记录采集时间;

因为前面文章包含了除RTC之外的所有的代码,此文章只放RTC代码。

二、工程源码

RTC.c

#include "sys.h"
#include "delay.h"
#include "usart.h"
#include "rtc.h" 		    //技术支持:_calendar_obj calendar;																										//时钟结构体 static void RTC_NVIC_Config(void)
{	NVIC_InitTypeDef NVIC_InitStructure;NVIC_InitStructure.NVIC_IRQChannel = RTC_IRQn;													//RTC全局中断NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;								//先占优先级0级NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;											//先占优先级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) != 0x5057)																	//从指定的后备寄存器中读出数据:读出了与写入的指定数据不相乎{	 			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(2024,9,13,16,00,00);  																								//设置时间	RTC_ExitConfigMode(); 																												//退出配置模式  BKP_WriteBackupRegister(BKP_DR1, 0X5057);																			//向指定的后备寄存器中写入用户程序数据}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_ClearITPendingBit(RTC_IT_SEC|RTC_IT_ALR);																		//清闹钟中断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寄存器的写操作完成  	RTC_Get();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);
}			  

RTC.h

#ifndef __RTC_H
#define __RTC_H	    //技术支持:
#include "sys.h"//时间结构体
typedef struct 
{vu8 hour;vu8 min;vu8 sec;			//公历日月年周vu16 w_year;vu8  w_month;vu8  w_date;vu8  week;		 
}_calendar_obj;	extern _calendar_obj calendar;																	//日历结构体extern u8 const mon_table[12];																	//月份日期数据表
void Disp_Time(u8 x,u8 y,u8 size);															//在制定位置开始显示时间
void Disp_Week(u8 x,u8 y,u8 size,u8 lang);											//在指定位置显示星期
u8 RTC_Init(void);        																			//初始化RTC,返回0,失败;1,成功;
u8 Is_Leap_Year(u16 year);																			//平年,闰年判断
u8 RTC_Get(void);         																			//更新时间   
u8 RTC_Get_Week(u16 year,u8 month,u8 day);
u8 RTC_Set(u16 syear,u8 smon,u8 sday,u8 hour,u8 min,u8 sec);		//设置时间	#endif

main.c


char time_buf[16]; 																								//记录时间//将时间转换为字符串void Time_To_String(void) 
{sprintf(time_buf, "%d/%d/%d|%02d:%02d:%02d",             calendar.w_year,calendar.w_month,calendar.w_date,calendar.hour, calendar.min,     calendar.sec);printf("time_buf: %s\r\n", time_buf);
}


http://www.mrgr.cn/news/25775.html

相关文章:

  • Linux(4)--CentOS8虚拟机下联网
  • JAVA学习-练习试用Java实现“将数据流变为多个不相交区间”
  • 决策树算法的介绍与应用
  • 5 存储器
  • 「2.1」前缀后缀——字符串哈希
  • 使用实时编辑器任务清理杂乱数据并定位极值
  • mysql DBA常用的sql
  • AttributeError: module ‘numpy‘ has no attribute ‘object‘.
  • C++奇迹之旅:快速上手Priority_queue的使用与模拟实现
  • FP7122:异步降压恒流LED驱动芯片
  • 022.PL-SQL进阶—分页过程
  • 如何带领一个兼职团队完成一个百万项目?
  • 【启扬方案】基于RK3588的割草机器人应用解决方案
  • C++继承
  • Leetcode面试经典150题-202.快乐数
  • 《小迪安全》学习笔记04
  • VSCode配置C++环境
  • 登山第九梯:稀疏点云实例分割——又快又准
  • 图片详解,最简单易懂!!!Ubuntu增强功能
  • 文章解读与仿真程序复现思路——电网技术EI\CSCD\北大核心《基于攻击者视角的综合能源系统网络攻击策略 》