你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

发布基于ANSI-C的RTC_Time库,利用UNIX时间戳格式,无中断实现万年历

[复制链接]
jjldc 发布时间:2009-1-4 16:19
基于STM32处理器
3 Q' {* F5 m  B, O6 h4 s! [RTC只是个能靠电池维持运行的32位定时器over!
: @" h$ Y6 M1 y并不像实时时钟芯片,读出来就是年月日。。。
' T- h5 v3 t% H5 y8 x0 B" o看过些网上的代码,有利用秒中断,在内存中维持一个年月日的日历。3 c" \1 Z7 A. K0 G" H4 R5 e6 ]9 I9 R6 l2 o
我觉得,这种方法有很多缺点:
7 I( L7 i2 J9 `0 C& o1.断电时没有中断可用: ^4 Y- L) y. \; W9 _; o. P7 w$ A
2.频繁进中断,消耗资源
2 ^3 S' s( k8 k  P3.时间运算复杂,代码需要自己写
$ M: `, G9 u# o3 J4.不与国际接轨。。。。
) T2 T4 r: R1 R! U
2 F% l6 m% d# s5 `8 f: R: {so,还是用标准的UNIX时间戳来进行时间的操作吧!# D6 G7 ^% f" t5 M+ y
什么是UNIX时间戳?
" I" W. h7 I# {! Z0 h0 c: k; RUNIX时间戳,是unix下的计时方式。。。很废话
% k3 @8 @. u: ?) @, W7 E具体点:他是一个32位的整形数(刚好和STM32的RTC寄存器一样大),表示从UNIX元年(格林尼治时间1970-1-1 0:0:0)开始到某时刻所经历的秒数2 ?1 L' q2 ^" v$ I1 ?1 [" s
听起来很玄幻的,计算下: 32位的数从0-0xFFFFFFFF秒,大概到2038年unix时间戳将会溢出!这就是Y2038bug
) y0 `; \5 v# h& L不过,事实上的标准,我们还是照这个用吧,还有二十年呢。。。
7 ?4 _( Z) Q/ s$ t' [. Q$ s
8 }4 a* @' ^, i7 jUNIX时间戳:1229544206  现实时间:2008-12-17  20:03:26 7 D; T5 y- h' I- A7 w
* _+ C" U: @# k9 T5 i  u# `' O6 t+ S
我们要做的,就是把当前时间的UNIX时间戳放在RTC计数器中让他每秒++,over
+ I% G* i1 F: L+ H1 M5 l: ]然后,设计一套接口函数,实现UNIX时间戳与年月日的日历时间格式转换 这样就可以了, T4 M4 d/ i0 r" {! ~( V) U

6 C& O, q. M1 C在RTC中实现这个时间算法,有如下好处:
' P1 ]8 q, S- R# o& s0 B1. 系统无需用中断和程序来维持时钟,断电后只要RTC在走即可; d! [0 D8 O6 A2 L/ {( M9 d7 w! Q$ M
2. 具体的两种计时的换算、星期数计算,有ANSI-C的标准C库函数实现,具体可以看time.h
% t. O/ B, ~/ P, s% g% p6 {3. 时间与时间的计算,用UNIX时间戳运算,就变成了两个32bit数的加减法
' e& U6 x4 I: W) ~! q4. 与国际接轨。。。5 M  C: O7 G- f  P9 }

. A6 i. y) b2 k0 n3 |2 H
3 Z- V% i. Q  X) P, `( c幸好是与国际接轨,我们有time.h帮忙,在MDK的ARM编辑器下有,IAR下也有4 G$ X& C9 N7 O' D
其中已经定义了两种数据类型:unix时间戳和日历型时间
; b. S* Y& D. N time_t:       UNIX时间戳(从1970-1-1起到某时间经过的秒数)
. q  V8 `( J' b) p* \+ S     typedef unsigned int time_t;
6 r3 n! w. M/ ?5 R 
* ~  I( b7 F$ Y3 z+ j; t" l6 D! v struct tm:    Calendar格式(年月日形式)# C' m1 M2 r7 C

+ `$ A& Q. d5 {. e同时有相关操作函数
. C/ s& s- L8 v- v) y7 F- T* _gmtime,localtime,ctime,mktime等等,方便的实现各种时间类型的转换和计算
" {3 P  C# s- o1 y! _- f4 D8 r. Q0 M; l% O& z" h3 |( X0 w9 U5 p( k
于是,基于这个time.h,折腾了一天,搞出了这个STM32下的RTC_Time使用的时间库8 ~0 K  P  i3 @/ _! k3 \# L

7 I8 q4 [; j; D9 t! c* {  B$ n1 z& o  a  Z
这是我的RTC_Time.c中的说明:. G) m/ I+ Q: `1 H. [
5 X5 m) Z, `9 {0 b9 U
 本文件实现基于RTC的日期功能,提供年月日的读写。(基于ANSI-C的time.h)
7 a3 l0 f: N7 F7 P # N  d+ k$ _& r; s; j
 作者:jjldc (九九)
2 o2 x* q" s8 R/ k5 x# { QQ: 77058617! U- d# \* q8 }
 
3 j* t2 c3 V3 I" v: i4 v RTC中保存的时间格式,是UNIX时间戳格式的。即一个32bit的time_t变量(实为u32)
% T# \9 Q9 ^3 ^1 Q( ^- c
; @) J) Y6 |* _% @ ANSI-C的标准库中,提供了两种表示时间的数据  型:: F. b2 F" Z7 c3 q8 y0 |- a
 time_t:       UNIX时间戳(从1970-1-1起到某时间经过的秒数)0 v' D! E3 f" S+ y8 c6 B$ a7 u
     typedef unsigned int time_t;% @1 |) ~$ A! A# F# Z
 
/ L0 u  B3 v% Y struct tm:    Calendar格式(年月日形式)
( k6 Q* x( z& M/ v7 n# y   tm结构如下:
- u' I/ i- _$ n6 j/ f7 V, e7 T   struct tm {
- \7 s! z5 {9 X6 ]       int tm_sec;   // 秒 seconds after the minute, 0 to 60
  \  U: Z; k& V% x  _4 x/ i6 S                        (0 - 60 allows for the occasional leap second), N0 |0 `* {, R/ k: C- o" g5 W5 t
       int tm_min;   // 分 minutes after the hour, 0 to 59
: Y' i1 J: U' s3 M8 t/ b        int tm_hour;  // 时 hours since midnight, 0 to 23' G! ~4 M4 ^+ E5 M' Y
        int tm_mday;  // 日 day of the month, 1 to 31
% W0 [$ ~% ~) ]( {' H        int tm_mon;   // 月 months since January, 0 to 11
: F; O7 D- m3 R% _) o3 `* u8 U  N6 f        int tm_year;  // 年 years since 19004 U" j/ i4 ]) W2 J& ~/ Q9 ^% a
        int tm_wday;  // 星期 days since Sunday, 0 to 64 {! U; \8 ?- [* M; i* w) Z4 w. }
        int tm_yday;  // 从元旦起的天数 days since January 1, 0 to 365; Z" z. i4 Q" s9 R5 ~/ {0 l
         int tm_isdst; // 夏令时??Daylight Savings Time flag0 q/ _% A3 }8 m5 N4 e3 i3 ?6 G7 w3 i) e
         ...
. ]% Z% G$ L" W* y+ M4 a     }
2 }. ^/ D+ J  c% X3 f! L8 Y     其中wday,yday可以自动产生,软件直接读取. X7 c; }' n9 H0 l% Z) V
     mon的取值为0-11
/ {: z& d1 \* L$ R" _% n# y8 O    ***注意***:
* A4 t! w1 V3 F. C5 B8 D& G1 F' B4 O    tm_year:在time.h库中定义为1900年起的年份,即2008年应表示为2008-1900=108
) {) v% S6 y1 P7 w+ Z5 A) e! ]# T     这种表示方法对用户来说不是十分友好,与现实有较大差异。5 ?% p6 I5 \* `# m
     所以在本文件中,屏蔽了这种差异。$ i4 ~+ |* k/ z. x
     即外部调用本文件的函数时,tm结构体类型的日期,tm_year即为2008" g7 j. o% n1 a2 A; ]; [
     注意:若要调用系统库time.c中的函数,需要自行将tm_year-=1900
, k( E7 h# @, M! Z8 V , I3 _! V* I* @8 M9 z
 成员函数说明:
4 u/ ]5 d* F7 M, n6 c+ }. `6 b struct tm Time_ConvUnixToCalendar(time_t t);; `6 {! p+ B; @: L$ h# e& E
     输入一个Unix时间戳(time_t),返回Calendar格式日期* C  Y: ~7 j2 ]+ `- \
 time_t Time_ConvCalendarToUnix(struct tm t);
1 U6 |# y  B+ O3 x+ X, P! ?% u     输入一个Calendar格式日期,返回Unix时间戳(time_t)
7 S9 k$ G( U  S: _; v( \3 g" C7 p6 n( x time_t Time_GetUnixTime(void);* R5 Z" E9 O, Y' Y
     从RTC取当前时间的Unix时间戳值6 v$ F% _  B6 `" B8 u! U6 W
 struct tm Time_GetCalendarTime(void);
7 m! ^7 @! z0 S+ e- U     从RTC取当前时间的日历时间
2 i- ]$ a0 w; a6 W& o, `5 Y" W5 \ void Time_SetUnixTime(time_t);
* y2 k/ v. B  g# d     输入UNIX时间戳格式时间,设置为当前RTC时间
5 y3 @* O+ N0 M% p' y$ t4 H. F void Time_SetCalendarTime(struct tm t);( h+ K0 S  Y" |/ M% `& k6 H" Z
     输入Calendar格式时间,设置为当前RTC时间- O& `) ~5 `3 M) G. g+ q$ \6 X, j
 
* q) {0 W+ {9 A0 x 外部调用实例:
$ {. P6 s8 ]3 ?9 O0 [% \+ q& @ 定义一个Calendar格式的日期变量:
$ w+ N3 l. W6 k% u5 C struct tm now;
* @  Y  ?- ~1 ?; m! N now.tm_year = 2008;
% O; G; {5 ?7 b  P& z now.tm_mon = 11;        //12月
6 a4 M# d6 W) H6 t# M* d$ h now.tm_mday = 20;; D5 V! q' g' z) c; h( @
 now.tm_hour = 20;- J/ ^+ c& w8 u/ c, j
 now.tm_min = 12;
5 s% D( L" I; U* Q- U/ L, e now.tm_sec = 30;5 y$ W% b- U' K' `5 s8 P7 \
 
% K5 W: D7 Q1 J" r. ^$ `# ?4 G/ [ 获取当前日期时间:) Q; y4 E7 u8 r3 d( M- h
 tm_now = Time_GetCalendarTime();: b) h2 T5 n' f+ y8 S& [! ?
 然后可以直接读tm_now.tm_wday获取星期数
0 u0 q" \5 b  a' R% ^7 g  s5 F0 ` 
, I+ V; A  C& e; B* S& W 设置时间:' y2 T0 f( Y9 _0 p6 V6 P
 Step1. tm_now.xxx = xxxxxxxxx;
7 C" W8 u1 A7 u2 h, M Step2. Time_SetCalendarTime(tm_now);
7 E: u  l" O: } - T# [- q7 p" _/ X7 j3 l
 计算两个时间的差
+ ~) T! w& ^! z8 C+ u struct tm t1,t2;& X5 f5 _; ?7 k
 t1_t = Time_ConvCalendarToUnix(t1);
; Z0 s1 K$ K: @7 H$ { t2_t = Time_ConvCalendarToUnix(t2);
9 r% q" T! W/ x dt = t1_t - t2_t;8 _% O3 V3 |3 a" f
 dt就是两个时间差的秒数
8 Y" K! W6 h5 G dt_tm = mktime(dt);    //注意dt的年份匹配,ansi库中函数为相对年份,注意超限' N3 a) U0 J/ W3 Y
 另可以参考相关资料,调用ansi-c库的格式化输出等功能,ctime,strftime等6 L' k! [; y! c! @& ]/ y
   }8 Q1 q, @* s  n
5 i, I) n; ]0 s2 n" b6 Z0 j
这是包含了RTC_Time的工程实例,可以用来参考
! F1 M% O! Z0 f6 r基于MDK环境. ^! P3 t! D' l$ ^( J2 Z6 b( ?
 9 m5 U: r1 X. O9 }. E3 @  G4 G8 m
 

发布基于ANSI-C的RTC_Time库,利用UNIX时间戳格式,无中断实现万年历.rar

下载

179.12 KB, 下载次数: 92

收藏 评论0 发布时间:2009-1-4 16:19

举报

0个回答

所属标签

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版