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

news2024/9/20 18:00:52

一、简介

实验目的:在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.coloradmin.cn/o/2130809.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

人工智能|集成学习——混合专家模型 (MoE)

随着 Mixtral 8x7B (announcement, model card) 的推出&#xff0c;一种称为混合专家模型 (Mixed Expert Models&#xff0c;简称 MoEs) 的 Transformer 模型在开源人工智能社区引起了广泛关注。在本篇博文中&#xff0c;我们将深入探讨 MoEs 的核心组件、训练方法&#xff0c;…

arcgisPro绘制平行线、垂直线段

1、绘制一条线 2、点击【创建要素】按钮&#xff0c;选择线&#xff0c;点一个点后&#xff0c;将鼠标移至需要对其的线上&#xff0c;并右击&#xff0c;选择【平行】 3、移动一段距离后&#xff0c;完成绘制&#xff0c;可得到一条平行线 4、得到平行线 5、绘制垂直线&#x…

一文读懂:区块链的原理、技术、应用领域

引言 在当今数字化时代&#xff0c;区块链技术已经成为全球范围内备受瞩目的话题。从金融到供应链&#xff0c;从物联网到数字身份&#xff0c;区块链正在以惊人的速度渗透到各个行业&#xff0c;并在重塑着我们的社会和经济格局。 区块链最初因其作为比特币背后技术的而引起…

OrionX vGPU研发测试场景下最佳实践之SSH模式

开发机场景概述 目前很多企业在做AI开发的场景时&#xff0c;对GPU资源的管理都是非常简单粗暴的。他们大多都是以开发小组为管理单位、由运维以台为单位分配给开发工程师使用。而在AI开发中涉及开发的场景和测试的场景&#xff0c;很多是将开发测试甚至训练任务都放在一起来使…

<<编码>> 第 11 章 逻辑门电路--门电路 示例电路

作为门电路的继电器开关电路 info::操作说明 鼠标单击开关切换开合状态 primary::在线交互操作链接 https://cc.xiaogd.net/?startCircuitLinkhttps://book.xiaogd.net/code-hlchs-examples/assets/circuit/code-hlchs-ch11-06-relay-as-gate.txt 作为反相器的继电器开关电路 …

EPON光模块介绍

EPON光模块在依靠光纤网络实现快速可靠的数据传输、增强带宽能力和提高网络效率的过程中发挥着至关重要的作用。在这篇文章中&#xff0c;我们将深入研究EPON光模块的基本概念、各种类型、优点和局限性&#xff0c;全面了解它们在现代电信中的重要性。 EPON光模块的定义 EPON…

ZYNQ 入门笔记(二):动态时钟

文章目录 1 概述1.1 DRP1.2 AXI4-Lite 2 示例2.1 单时钟输出2.2 多时钟输出 3 参考文档 1 概述 Clocking Wizard 可通过配置内部寄存器动态调整输出频率&#xff0c;配置接口可选 DRP 或 AXI4-Lite&#xff0c;其中 AXI4-Lite 实际上是对 DRP 接口的封装 1.1 DRP 通过 DRP 接…

Python文件操作:上下文管理器(with语句)②

文章目录 1. 上下文管理器概述1.1 什么是上下文管理器&#xff1f;1.2 为什么使用上下文管理器&#xff1f;1.3 with语句的基本语法 2. 文件操作中的上下文管理器2.1 使用with语句打开文件2.2 读取文件2.2.1 读取整个文件内容2.2.2 逐行读取文件 2.3 写入文件2.3.1 覆盖写入2.3…

JavaWeb使用web.xml配置Servlet路径映射的相关操作以及易错问题分析与解决

前言 我们在使用Servlet创建JavaWeb项目时&#xff0c;想要绑定url路径和Servlet的映射关系&#xff0c;需要在web.xml中配置映射关系。Servlet从2.5版本开始支持注解。具体来说&#xff0c;Servlet 2.5引入了注解配置方式&#xff0c;使得Servlet应用程序的配置更加简单、灵活…

文档智能:OCR+Rocketqa+layoutxlm

此次先记录LayoutLMv2&#xff0c;梳理相关论文&#xff0c;记录如下&#xff1a; 首先认识一下 visually-rich document understanding tasks → \to → VrDU 其次&#xff0c;the text fields of interest&#xff0c;与图像识别的感兴趣区域 region of Interest 类似&…

MySQL数据的增删改查(一)

目录 新增&#xff08;create&#xff09; 插入单条记录 插入多条记录 查询&#xff08;retrieve&#xff09; 查询所有列 查询特定列 查询字段为表达式 别名 去重 排序 按单列排序 按多列排序 使用表达式或别名排序 排序NULL值 条件查询 比较运算符 逻辑运算…

双向dfs,多次dfs

前言&#xff1a;这个答案给我们提供了一种多次dfs的思路&#xff0c;记录queue的size&#xff0c;每次只取size个&#xff0c;就刚刚好只处理了上一次的‘ 题目地址 #include<bits/stdc.h> using namespace std;//定义队列节点 struct node {int x,y; }rear,front; //Q[…

Leetcode面试经典150题-350.两个数组的交集II

题目比较简单&#xff0c;散散心吧 这个题竟然是349更简单的版本&#xff0c;可以先看看349题的解法&#xff1a; Leetcode面试经典150题-349.两个数组的交集-CSDN博客 349会了&#xff0c;这个也就会了 解法都在代码里&#xff0c;不懂就留言或者私信 class Solution {/…

C51单片机矩阵键盘输入数码管静态显示

【实验目的】 学会矩阵键盘的检测&#xff0c;掌握数码管静态显示原理。 【实验现象】 依次按下4*4矩阵键盘上从第1到第20个键&#xff0c;同时在六位数码管上依次显示0、1、2、3、4、5、6、7、8、9、A、B、C、D、E、F。 【实验说明】 本开发板上数码管为共阴极。静态数码管显示…

R与机器学习系列|15.可解释的机器学习算法(Interpretable Machine Learning)(下)

今天我们介绍可解释机器学习算法的最后一部分&#xff0c;基于XGBoost算法的SHAP值可视化。关于SHAP值其实我们之前的很多个推文中都介绍到&#xff0c;不论是R版本的还是Python版本的&#xff0c;亦不论是普通的分类问题还是生存数据模型的。在此推文中我们将基于XGBoost模型理…

付费进群付费入群流量掘金入群系统九牧云版源码系统搭建

适用于各类资源类付费进群领取&#xff0c;私域类项目经营等 简洁大气直观。流量掘金类必备。 前端展示视频&#xff1a; https://pan.baidu.com/s/1lqyGCOrfmE4LDXb1cm-eDQ?pwdvnk6 https://yun.ktbf.xyz/s/by6jIzghpb 大致功能&#xff1a; 支持域名防红模式 支持对接…

QT+OSG+OSG-earth如何在窗口显示一个地球

1、环境配置 系统&#xff1a;windows10系统 QT:版本5.15.2 编译器&#xff1a;MSVC2019_64bit 编辑器&#xff1a;QT Creator OSG版本&#xff1a;3.7.0 64位 为MSVC环境下编译 osgQt:为第三方编译的库&#xff0c;OSG因为版本不同已经不提供osgQt的…

【一文就懂】计算机视觉期刊和会议缩写

下面IEEE相关的期刊及其缩写&#xff0c;并重新整理为期刊和会议两个部分。 期刊缩写 期刊全称缩写IEEE Transactions on Pattern Analysis and Machine IntelligenceIEEE Trans. Pattern Anal. Mach. Intell.IEEE Transactions on Image ProcessingIEEE Trans. Image Proce…

用于大数据分析的数据存储格式:Parquet、Avro 和 ORC 的性能和成本影响

高效的数据处理对于依赖大数据分析做出明智决策的企业和组织至关重要。显著影响数据处理性能的一个关键因素是数据的存储格式。本文探讨了不同存储格式&#xff08;特别是 Parquet、Avro 和 ORC&#xff09;对 Google Cloud Platform &#xff08;GCP&#xff09; 上大数据环境…

机器学习--支持向量机(SVM)

支持向量机(线性) S V M SVM SVM 引入 S V M SVM SVM 用于解决的问题也是 c l a s s i f i c a t i o n classification classification&#xff0c;这里 y ∈ { − 1 , 1 } y \in \{-1, 1\} y∈{−1,1} 比如说这样一个需要分类的训练数据&#xff1a; 我们可以有很多直线来…