HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
time.c
Go to the documentation of this file.
1 
25 #include "user_config.h"
26 
27 #ifdef AVR
28 #include <stdlib.h>
29 #include <string.h>
30 #endif
31 
32 #include "fatfs.h"
33 
34 #include "mathio.h"
35 
36 #include "time.h"
37 #include "timer.h"
38 
39 #ifdef RTC_SUPPORT
40 #include "rtc.h"
41 #endif
42 
43 #include "posix.h"
44 
46 extern volatile ts_t __clock;
47 
50 
53 
59 static const uint16_t __days_sum[] =
60 {
61  0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365
62 };
63 
68 static const uint16_t __days[] =
69 {
70  31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31
71 };
72 
78 const char *__WDay[] = { "Sun","Mon","Tue","Wed","Thu","Fri","Sat","BAD"};
79 
87 char *tm_wday_to_ascii(int i)
88 {
89  if(i >= 0 && i <= 6)
90  return((char *)__WDay[i]);
91  else
92  return((char *)__WDay[7]);
93 }
94 
95 
100 const char *__Month[]= \
101 { \
102  "Jan","Feb","Mar","Apr","May","Jun","Jul", \
103  "Aug","Sep","Oct","Nov","Dec","BAD"
104 };
105 
113 MEMSPACE
114 char *tm_mon_to_ascii(int i)
115 {
116  if(i >= 0 && i <= 11)
117  return((char *)__Month[i]);
118  else
119  return((char *)__Month[12]);
120 }
121 
122 
134 MEMSPACE
135 static int IS_Leap(int year)
136 {
137  if((year & 3) || year == 1900 || year == 2100)
138  return(0);
139  return(1);
140 }
141 
142 
155 MEMSPACE
156 static int Leap_Days_Since_1900(int year)
157 {
158  int sum;
159 
160  year -= 1900;
161 
162  if(year>0)
163  --year;
164 
165  sum = (year >> 2);
166 
167  if(year >= 200 )
168  --sum;
169 
170  return(sum);
171 }
172 
173 
179 MEMSPACE
180 int finddayofweek(int year, int month, int day)
181 {
182  int value;
183 
184  if ( month < 3 )
185  {
186  month += 12;
187  year -= 1;
188  }
189  value = ( year + year/4 - year/100 + year/400 );
190  value += ( 6 * (month+1) / 10 + (month * 2));
191  value += day;
192  value += 1;
193  return( value % 7 );
194 }
195 
196 
209 MEMSPACE
210 static int Days_Per_Year(int year)
211 {
212  return(IS_Leap(year) ? 366 : 365);
213 }
214 
215 
222 MEMSPACE
223 int Days_Per_Month(int month, int year)
224 {
225  int days;
226 
227 // Normalize month
228  while(month >= 12)
229  {
230  ++year;
231  month -= 12;
232  }
233  while(month < 0)
234  {
235  --year;
236  month += 12;
237  }
238  days = __days[month];
239  if( month == 1 && IS_Leap(year))
240  ++days;
241  return( days );
242 }
243 
244 
254 
261 
262 MEMSPACE
263 time_t time_to_tm(time_t epoch, int32_t offset, tm_t *t)
264 {
265  int year,month,tmp;
266  int flag = 0;
267  int32_t days;
268  time_t save = epoch;
269 
270  memset(t,0,sizeof(tm_t));
271 
272  if(epoch >= 0xFFFD5D00UL)
273  return(-1);
274 
275  epoch -= offset;
276 
277  if(epoch >= 0xFFFEAE80UL)
278  {
279  epoch -= 0xFFFEAE80UL;
280  flag = 1;
281  }
282 
283  t->tm_sec = epoch % 60;
284  epoch /= 60;
285 
286  t->tm_min = epoch % 60;
287  epoch /= 60;
288 
289  t->tm_hour = epoch % 24;
290  epoch /= 24;
291 
292  days = epoch;
293 
294  if(flag)
295  {
296  t->tm_year = 69;
297  t->tm_mon = 11;
298  t->tm_mday = 31;
299  t->tm_wday = (EPOCH_DAY - 1) % 7;
300  }
301  else
302  {
303  t->tm_wday = (EPOCH_DAY + days) % 7;
304 
305  year=EPOCH_YEAR;
306  while (days >= (tmp=Days_Per_Year(year)) )
307  {
308  ++year;
309  days -= tmp;
310  }
311 
312  t->tm_year = year - 1900;
313  t->tm_yday = days;
314 
315  month = 0;
316 
317  while(days > 0 && month < 12)
318  {
319  tmp=__days[month];
320  if( month == 1 && IS_Leap(year))
321  ++tmp;
322  if(days < tmp)
323  break;
324  days -= tmp;
325  ++month;
326  }
327 
328  t->tm_mon = month;
329  t->tm_mday = days + 1;
330  }
331  return(save - offset);
332 }
333 
334 
347 MEMSPACE
349 {
350  time_t seconds;
351  seconds = normalize(t,0);
352  return (seconds);
353 }
354 
355 
356 // =============================================
358 // =============================================
359 
367 MEMSPACE
368 char *asctime_r(tm_t *t, char *buf)
369 {
370 // normaize tm_t before output
371  (void) normalize(t,0);
372 
373  memset(buf,0,32);
374  snprintf(buf,32,"%s %s %2d %02d:%02d:%02d %4d",
375  __WDay[t->tm_wday],
376  __Month[t->tm_mon],
377  (int)t->tm_mday,
378  (int)t->tm_hour,
379  (int)t->tm_min,
380  (int)t->tm_sec,
381  (int)t->tm_year + 1900);
382  return(buf);
383 }
384 
385 
393 MEMSPACE
394 char *asctime(tm_t *t)
395 {
396  static char buf[32];
397 // acstime_r does tm_t normalization
398  return( asctime_r(t,buf) );
399 }
400 
401 
409 MEMSPACE
410 char *ctime_r(time_t *t, char *buf)
411 {
412 // acstime_r does tm_t normalization
413  return( asctime_r(localtime(t),buf) );
414 }
415 
416 
423 MEMSPACE
424 char *ctime(time_t *tp)
425 {
426  static char buf[32];
427 // acstime_r does tm_t normalization
428  return( asctime_r( localtime(tp), buf) );
429 }
430 
431 
440 MEMSPACE
441 char *ctime_gm(time_t *tp)
442 {
443  static char buf[32];
444  tm_t tm;
445  return( asctime_r( gmtime_r(tp,&tm), buf) );
446 }
447 
448 
455 MEMSPACE
456 tm_t *gmtime_r(time_t *t, tm_t *result)
457 {
458  time_t epoch = *t;
459  (void)time_to_tm(epoch, 0, result);
460  return(result);
461 }
462 
463 
470 MEMSPACE
472 {
473  static tm_t t, *p;
474  p = &t;
475  gmtime_r(tp, p);
476  return(p);
477 }
478 
479 
480 // FIXME localtime
481 /*
482  The localtime function converts the calendar time timep to broken-down time representation, expressed relative to the users specified timezone.
483  The function acts as if it called tzset(3) and sets the external variables tzname with information about the current timezone, timezone with the difference
484  between Coordinated Universal Time (UTC) and local standard time in seconds, and daylight to a nonzero value if daylight savings time rules apply during
485  some part of the year. The return value points to a statically allocated struct which might be overwritten by subsequent calls to any of the date and
486  time functions. The localtime_r() function does the same, but stores the data in a user-supplied struct. It need not set tzname, timezone, and day� light. */ /// @brief Convert POSIX epoch time_t *tp into POSIX tm_t *result /// expressed as local time using timezone and DST corrections. /// /// @param[in] t: time_t * epoch time input. /// @param[out] result: tm_t *result. /// /// @return result. MEMSPACE tm_t *localtime_r(time_t *t, tm_t *result) { tz_t tz; long int offset; time_t epoch = *t; gettimezone(&tz); offset = 60L * tz.tz_minuteswest; if(is_dst(epoch - offset)) offset -= 3600L; (void) time_to_tm(epoch, offset, result); return(result); } /// @brief Convert POSIX epoch time_t *tp into POSIX tm_t *result. /// /// @param[in] tp: time_t * epoch time input. /// /// @return struct tm result. /// @warning result is overwritten on each call. MEMSPACE tm_t *localtime(time_t *tp) { static struct tm t; return( localtime_r(tp, &t) ); } /// @brief convert tm_t structure to time_t local time epoch /// /// @param[in] t: tm_t time input. /// /// @return local epoch or -1 on failure out of range MEMSPACE time_t mktime(tm_t *t) { return( normalize(t, 1) ); } /// @brief Converts tm_t structure as GMT time into GMT epoch since 1900. /// - used internally by normalize() function after normalization /// - DO NOT use in user code , use normaize or any function that uses it /// /// - Standards: none /// - Limits: year(1900..2199). /// - Assume: epoch size is time_t; /// /// @return Seconds since GMT EPOCH_YEAR Jan 1st. /// @return -1 on error. MEMSPACE static time_t tm2epoch( tm_t *t ) { time_t days,seconds; int year = t->tm_year + 1900; int mon = t->tm_mon; // 0..11 int mday = t->tm_mday - 1; // 1..28,29,30,31 int hour = t->tm_hour; // 0..23 int min = t->tm_min; // 0..59 int sec = t->tm_sec; // 0..59 #ifdef TIME_DEBUG printf("tm2epoch %4d,%2d,%2d, %02d:%02d:%02d\n", (int)t->tm_year+1900, (int)t->tm_mon, (int)t->tm_mday, (int)t->tm_hour, (int)t->tm_min, (int)t->tm_sec); #endif if (year < EPOCH_YEAR || year > 2106) { #ifdef TIME_DEBUG printf("tm2epoch year out of range: %4d\n", year); #endif return(-1); } if(mon >= 12 || mon < 0) { #ifdef TIME_DEBUG printf("tm2epoch mon out of range: %4d\n", mon); #endif return(-1); } if(mday >= Days_Per_Month(mon,year) || mday < 0) { #ifdef TIME_DEBUG printf("tm2epoch mday out of range: %4d\n", mday); #endif return(-1); } if(hour >= 24 || hour < 0) { #ifdef TIME_DEBUG printf("tm2epoch hour out of range: %4d\n", hour); #endif return(-1); } if(min >= 60|| min < 0) { #ifdef TIME_DEBUG printf("tm2epoch min out of range: %4d\n", min); #endif return(-1); } if(sec >= 60 || sec < 0) { #ifdef TIME_DEBUG printf("tm2epoch sec out of range: %4d\n", sec); #endif return(-1); } /// Note: To simplify we caculate Leap Day contributions in stages days = ( year - EPOCH_YEAR ); days *= (time_t) 365L; days += (time_t) __days_sum[mon]; days += (time_t) mday; days += (time_t) Leap_Days_Since_1900(year); days -= (time_t) 17; if(mon > 1 && IS_Leap(year)) ++days; seconds = days; seconds *= (time_t) 24L; // hours seconds += (time_t) hour; seconds *= (time_t) 60L; // Minutes seconds += (time_t) min; seconds *= (time_t) 60L; // Seconds seconds += (time_t) sec; return (seconds); } /// @brief Normalize POSIX tm_t *t struct and convert to epoch time /// Note: does not deal with DST - by design /// /// @param[in] t: tm_t time input. /// @param[in] normalize_to_timezone: nonzero = adjust to local timezone and DST /// /// @return epoch since 1900 on sucess, -1 on failure out of range MEMSPACE time_t normalize(tm_t *t, int normalize_to_timezone) { time_t epoch; int32_t offset; int isdst; // struct tm // { // int tm_sec; /*< Seconds. [0-60] (1 leap second) */ // int tm_min; /*< Minutes. [0-59] */ // int tm_hour; /*< Hours. [0-23] */ // int tm_mday; /*< Day. [1-31] */ // int tm_mon; /*< Month. [0-11] */ // int tm_year; /*< Year - 1900. */ // int tm_wday; /*< Day of week. [0-6] */ // int tm_yday; /*< Days in year.[0-365] */ // int tm_isdst; /*< DST. [-1/0/1] */ // }; // Normalize t->tm_seconds while(t->tm_sec >= 60) { ++t->tm_min; t->tm_sec -= 60; } while(t->tm_sec < 0) { --t->tm_min; t->tm_sec += 60; } // Normalize t->tm_miniutes while(t->tm_min >= 60) { ++t->tm_hour; t->tm_min -= 60; } while(t->tm_min < 0) { --t->tm_hour; t->tm_min += 60; } // Normalize t->tm_hours while(t->tm_hour >= 24) { ++t->tm_mday; t->tm_hour -= 24; } while(t->tm_hour < 0) { --t->tm_mday; t->tm_hour += 24; } // Normalize t->tm_months while(t->tm_mon >= 12) { ++t->tm_year; t->tm_mon -= 12; } while(t->tm_mon < 0) { --t->tm_year; t->tm_mon += 12; } // Normalize t->tm_mday // t->tm_mday is 1 based while(t->tm_mday > Days_Per_Month(t->tm_mon,t->tm_year) ) { // subtract days in current t->tm_month t->tm_mday -= Days_Per_Month(t->tm_mon,t->tm_year); // Keep month normalized if(++t->tm_mon >= 12) { t->tm_mon -= 12; ++t->tm_year; } } // t->tm_mday is 1 based while(t->tm_mday < 1) { // Keep month normalized if(--t->tm_mon < 0) { t->tm_mon += 12; --t->tm_year; } // add days in previous mount t->tm_mday += Days_Per_Month(t->tm_mon,t->tm_year); } // We can now set the remain values by converting to EPOCH and back again // convert to EPOCH based seconds epoch = tm2epoch( t ); #ifdef TIME_DEBUG printf("%10lu - epoch-1\n",epoch); #endif // Normaize to local timezone with DST offset = 0L; isdst = 0; if(normalize_to_timezone) { tz_t tz; gettimezone(&tz); offset = tz.tz_minuteswest * 60L; if(is_dst(epoch)) { offset -= 3600L; isdst = 1; } } #ifdef TIME_DEBUG printf("%10lu - epoch-2\n",epoch); #endif // convert back to TM structure epoch = time_to_tm(epoch, offset, t); if(isdst) t->tm_isdst = 1; #ifdef TIME_DEBUG printf("%10lu - epoch-3\n",epoch); #endif return( epoch ); } /// @brief Get current timezone in struct timezone *tz - POSIX function. /// /// @param[out] tz: timezone result. /// /// @return 0 MEMSPACE int gettimezone(tz_t *tz) { tz->tz_minuteswest = __tzone.tz_minuteswest; tz->tz_dsttime = __tzone.tz_dsttime; return(0); } /// @brief Set current timezone with struct timezone *tz - POSIX function. /// /// @param[in] tz: timezone result. /// /// @return 0. MEMSPACE int settimezone(tz_t *tz) { __tzone.tz_minuteswest = tz->tz_minuteswest; __tzone.tz_dsttime = tz->tz_dsttime; return(0); } /// @brief Get current time struct timeval *tv and struct timezone *tz - POSIX function. /// We assume a GMT hardware clock /// /// @param[in] tv: time. /// @param[in] tz: timezone. /// /// @return 0 MEMSPACE int gettimeofday(tv_t *tv, tz_t *tz) { ts_t ts; clock_gettime(0, (ts_t *) &ts); tv->tv_sec = ts.tv_sec; tv->tv_usec = ts.tv_nsec / 1000L; gettimezone(tz); return(0); } /// @brief Return second from epoch - POSIX function. /// /// @param[in,out] t: pointer to store time in. /// - Notes: If t is non-NULL, store the return value there also. /// @return time_t seconds from epoch. /// @see clock_gettime(). MEMSPACE time_t time(time_t *t) { static ts_t ts; clock_gettime(0, (ts_t *) &ts); if(t != NULL) *t = ts.tv_sec; return(ts.tv_sec); } /// @brief Set current time struct timeval *tv and struct timezone *tz - POSIX function. /// We assume a GMT hardware clock /// /// @param[in] tv: time. /// @param[in] tz: timezone. /// /// @return 0 MEMSPACE int settimeofday(tv_t *tv, tz_t *tz) { ts_t ts; ts.tv_sec = tv->tv_sec; ts.tv_nsec = tv->tv_usec * 1000L; clock_settime(0, (ts_t *) &ts); settimezone(tz); return(0); } /// @brief Set system clock with seconds and microseconds. /// /// - Note: Alternate clock functions. /// @return void. /// @see clock_settime(). MEMSPACE void clock_set(uint32_t seconds, uint32_t us) { ts_t ts; ts.tv_sec = seconds; ts.tv_nsec = us * 1000L; //FIXME we may want to add hooks to the OS time functions if they exist clock_settime(0, (ts_t *) &ts); } ///@brief Set date and time by prompting user. /// /// - Prompt use for Date Time input with "YYYY MM DD HH:MM:SS>" /// - Input format is: "YYYY MM DD HH:MM:SS" /// ///@return 0 on success. ///@return -1 on error> MEMSPACE int setdate (void) { char buf[40]; extern MEMSPACE char *fgets ( char *str , int size , FILE *stream ); printf("Enter date YYYY MM DD HH:MM:SS >"); fgets(buf,sizeof(buf)-2,stdin); return(setdate_r(buf)); } ///@brief Set date and time from string in this format "YYYY MM DD HH:MM:SS". /// ///@param[in] buf: Date string in this format "YYYY MM DD HH:MM:SS". /// ///@return 0 on success. ///@return(-1) on error. MEMSPACE int setdate_r (char *buf) { tm_t tm; ts_t ts; time_t seconds; tm.tm_year=tm.tm_mon=tm.tm_mday=tm.tm_hour=tm.tm_min=tm.tm_sec=0; #ifdef SMALL_SSCANF sscanf(buf,"%d %d %d %d:%d:%d", &tm.tm_year, &tm.tm_mon, &tm.tm_mday, &tm.tm_hour, &tm.tm_min, &tm.tm_sec); #else // Year while(*buf == ' ') ++buf; tm.tm_year = strtol(buf,&buf,10); // Month while(*buf == ' ') ++buf; tm.tm_mon = strtol(buf,&buf,10); // Day of Month while(*buf == ' ') ++buf; tm.tm_mday = strtol(buf,&buf,10); // Hour while(*buf == ' ') ++buf; tm.tm_hour = strtol(buf,&buf,10); // Minute if(*buf && (*buf == ' ' || *buf == ':')) ++buf; tm.tm_min = strtol(buf,&buf,10); // Second if(*buf && (*buf == ' ' || *buf == ':')) ++buf; tm.tm_sec = strtol(buf,&buf,10); #endif tm.tm_mon--; if(tm.tm_year < 1970 || tm.tm_year > 2038) { printf("invalid year: %d\n",tm.tm_year); return(-1); } if(tm.tm_year >= 1900) tm.tm_year -= 1900; if(tm.tm_mon < 0 || tm.tm_mon > 11) { printf("invalid mon: %d\n",tm.tm_year); return(-1); } if(tm.tm_mday < 1 || tm.tm_mday > 31) { printf("invalid day: %d\n",tm.tm_mday); return(-1); } if(tm.tm_hour < 0 || tm.tm_hour > 23) { printf("invalid hour: %d\n",tm.tm_hour); return(-1); } if(tm.tm_min < 0 || tm.tm_min > 59) { printf("invalid min: %d\n",tm.tm_min); return(-1); } seconds = timegm(&tm); ts.tv_sec = seconds; ts.tv_nsec = 0L; clock_settime(0, (ts_t *) &ts); #ifdef RTC_SUPPORT if( !rtc_init(1, (time_t) seconds ) ) { printf("rtc force init failed\n"); return(-1); } #endif return(0); } /// @brief Calculate GMT seconds of DST transition given LOCAL time start / end time and DST flag /// /// @param[in] dst: 0 .. 1 DST needs to be applied to the arguments for DST caluclulations /// @param[in] epoch: 0 | epoch if non-zero - UTC epoch time used to obtain year of DST calculations /// @param[in] year: 0 | year if non-zero - UTC year of DST calcululation, if year and epoch are used - ignore epoch /// @param[in] month: 0 .. 11, local time month DST transition /// @param[in] weekno: 1 .. 4 localtime dayno count in this month /// @param[in] dayno: 0 .. 6, localtime day of DST transition, 0 = Sunday /// @param[in] hour: 0 .. 23 local time hour of DST transition /// result is in epoch GMT seconds /// Example: Eastern Time and Daylight Savings time /// time_t epoch,start,end; /// tv_t tv; /// tv_t tz; /// gettimeofday(&tv, &tz); /// epoch = tv.tv_sec; /// // DST start for this year, 2nd Sunday of Mar at 2:00am EST /// start = find_dst(0, epoch, 0, 3, 2, 0, 2); /// // DST start on 2016, 2nd Sunday of Mar at 2:00am EST /// start = find_dst(0, 0, 2016, 3, 2, 0, 2); /// // DST ends on for this year, 1st Sunday of Nov at 2:00am DST /// end = find_dst(1, epoch, 0, 11, 1, 0, 2); /// // DST ends on 2016, 1st Sunday of Nov at 2:00am DST /// end = find_dst(1, 0, 2016, 11, 1, 0, 2); MEMSPACE time_t find_dst(int dst, time_t epoch, int year, int month, int weekno, int dayno, int hour) { tm_t t; tz_t tz; tv_t tv; time_t dst_epoch; int32_t offset; int days; // Get timezone and clock time gettimeofday(&tv, &tz); // Local time offset in seconds // Get local timezone offset in seconds without DST offset offset = tz.tz_minuteswest * 60UL; // Add DST offset if DST end time includes DST offset if(dst) offset -= (60UL * 60UL); if(year) { // Year is specified - use it t.tm_year = year - 1900; // 0 = 1900 } else { // Otherwise, Calculate year from epoch or GMT time if(!epoch) epoch = tv.tv_sec; (void) time_to_tm(epoch, offset, &t); } // Local time of DST // We compute seconds for start of month in which DST happens // Result from timegm() is in GMT t.tm_mon = month - 1; // 0..11 t.tm_mday = 1; // 1..28,29,30,31 t.tm_hour = hour; // 0 .. 23 t.tm_min = 0; // 0..59 t.tm_sec = 0; // 0..59 // Adjust tm_t to localtime - we can not use normalize as it calls us // local seconds as timegm does not add offset dst_epoch = timegm(&t); // Add offset for localtime dst_epoch += offset; // Convert to back to tm_t structure with localtime offsets applied (void) time_to_tm(dst_epoch, 0L, &t); dayno = dayno; weekno= weekno; days= 0; while( 1 ) { if( ((t.tm_wday + days) % 7) == dayno) { if( (--weekno) == 0) break; } ++days; dst_epoch += 86400L; } // Return GMT return(dst_epoch); } /// @brief Set DST start and end time for the given epoch year /// @param[in] 0 - or epoch seconds in GMT used to determin the year to aply DST in /// If 0 we get local GMT epoch time in seconds MEMSPACE void set_dst(time_t epoch) { if(epoch == 0) { tv_t tv; tz_t tz; // Get timezone and clock time gettimeofday(&tv, &tz); epoch = tv.tv_sec; } // Cache caclulations so we do not do them more often then once a day if(dst.epoch >= epoch) { if((dst.epoch - epoch) < 86400L) return; } else { if((epoch - dst.epoch) < 86400L) return; } dst.epoch = epoch; // FIXME think of ways to cache this computation dst.start = find_dst(0, epoch, 0, 3, 2, 0, 2); dst.end = find_dst(1, epoch, 0, 11, 1, 0, 2); } /// @brief Test GMT epoch time to see if DST applies in a local timezone /// @param[in] epoch seconds in GMT /// @See normalize() /// return 1 if yes, 0 if not MEMSPACE int is_dst(time_t epoch) { set_dst(epoch); if( epoch >= dst.start && epoch <= dst.end) return(1); return(0); } /// @brief print start/stop for DST as localtime for this year MEMSPACE void print_dst() { printf("DST START localtime: %s\n", ctime(&dst.start)); printf("DST END localtime: %s\n", ctime(&dst.end)); } /// @brief print start/stop for DST as GMT for this year MEMSPACE void print_dst_gmt() { printf("DST START GMT: %s\n", ctime_gm(&dst.start)); printf("DST END GMT: %s\n", ctime_gm(&dst.end)); } /// @brief initialize system time - if we have an RTC use it /// @param[in] minwest: your time zone as minute west ///@return void ///@see: clock_settime() MEMSPACE void initialize_clock(int minwest) { time_t seconds = 0; tm_t tc; ts_t ts; tz_t tz; #ifdef RTC_SUPPORT if(!rtc_init(0,0L)) { printf("rtc uninitilized\n"); printf("attempting rtc init\n"); if( !rtc_init(1, (time_t) 0) ) { printf("rtc force init failed\n"); } } if(rtc_read(&tc)) { seconds = timegm(&tc); } else { seconds = 0; printf("rtc read errorafter init\n"); } #else printf("NO RTC\n"); seconds = 0; #endif // RTC_SUPPORT if(!seconds) printf("use setdate command to change time\n"); tz.tz_minuteswest = minwest; tz.tz_dsttime = 0; settimezone( &tz ); ts.tv_sec = seconds; ts.tv_nsec = 0L; clock_settime(0, (ts_t *) &ts); } /// @brief Display system time and optionally RTC time /// /// @return void /// @see: rtc_read /// @see: timegm() /// @see: ascitime() MEMSPACE void display_clock() { time_t seconds; tm_t tc; ts_t ts; #ifdef RTC_SUPPORT if(rtc_read(&tc)) { seconds = timegm(&tc); printf("rtc seconds: %lu\n",seconds); printf("rtc time: %s\n",asctime(&tc)); #if RTC_DEBUG printf("rtc_read:%d, day:%d,mon:%d,hour:%d,min:%d,sec:%d, wday:%d\n", (int) tc.tm_year + 1900, (int) tc.tm_mday, (int) tc.tm_mon, (int) tc.tm_hour, (int) tc.tm_min, (int) tc.tm_sec, (int) tc.tm_wday); #endif } else { printf("RTC read failed\n"); } #endif // RTC_SUPPORT clock_gettime(0, (ts_t *) &ts); seconds = ts.tv_sec; printf("clk seconds: %lu\n",seconds); printf("clk time: %s\n", asctime(gmtime(&seconds))); }
487  light.
488 */
489 
497 MEMSPACE
499 {
500  tz_t tz;
501  long int offset;
502  time_t epoch = *t;
503 
504  gettimezone(&tz);
505  offset = 60L * tz.tz_minuteswest;
506 
507  if(is_dst(epoch - offset))
508  offset -= 3600L;
509  (void) time_to_tm(epoch, offset, result);
510 
511  return(result);
512 }
513 
514 
521 MEMSPACE
523 {
524  static struct tm t;
525  return( localtime_r(tp, &t) );
526 }
527 
528 
534 MEMSPACE
536 {
537  return( normalize(t, 1) );
538 }
539 
540 
551 MEMSPACE
552 static
554 {
555  time_t days,seconds;
556 
557  int year = t->tm_year + 1900;
558  int mon = t->tm_mon; // 0..11
559  int mday = t->tm_mday - 1; // 1..28,29,30,31
560  int hour = t->tm_hour; // 0..23
561  int min = t->tm_min; // 0..59
562  int sec = t->tm_sec; // 0..59
563 
564 #ifdef TIME_DEBUG
565  printf("tm2epoch %4d,%2d,%2d, %02d:%02d:%02d\n",
566  (int)t->tm_year+1900, (int)t->tm_mon, (int)t->tm_mday,
567  (int)t->tm_hour, (int)t->tm_min, (int)t->tm_sec);
568 #endif
569 
570  if (year < EPOCH_YEAR || year > 2106)
571  {
572 #ifdef TIME_DEBUG
573  printf("tm2epoch year out of range: %4d\n", year);
574 #endif
575  return(-1);
576  }
577 
578  if(mon >= 12 || mon < 0)
579  {
580 #ifdef TIME_DEBUG
581  printf("tm2epoch mon out of range: %4d\n", mon);
582 #endif
583  return(-1);
584  }
585 
586  if(mday >= Days_Per_Month(mon,year) || mday < 0)
587  {
588 #ifdef TIME_DEBUG
589  printf("tm2epoch mday out of range: %4d\n", mday);
590 #endif
591  return(-1);
592  }
593 
594  if(hour >= 24 || hour < 0)
595  {
596 #ifdef TIME_DEBUG
597  printf("tm2epoch hour out of range: %4d\n", hour);
598 #endif
599  return(-1);
600  }
601 
602  if(min >= 60|| min < 0)
603  {
604 #ifdef TIME_DEBUG
605  printf("tm2epoch min out of range: %4d\n", min);
606 #endif
607  return(-1);
608  }
609 
610  if(sec >= 60 || sec < 0)
611  {
612 #ifdef TIME_DEBUG
613  printf("tm2epoch sec out of range: %4d\n", sec);
614 #endif
615  return(-1);
616  }
617 
619 
620  days = ( year - EPOCH_YEAR );
621 
622  days *= (time_t) 365L;
623 
624  days += (time_t) __days_sum[mon];
625 
626  days += (time_t) mday;
627 
628  days += (time_t) Leap_Days_Since_1900(year);
629 
630  days -= (time_t) 17;
631 
632  if(mon > 1 && IS_Leap(year))
633  ++days;
634 
635  seconds = days;
636 
637  seconds *= (time_t) 24L; // hours
638  seconds += (time_t) hour;
639  seconds *= (time_t) 60L; // Minutes
640  seconds += (time_t) min;
641  seconds *= (time_t) 60L; // Seconds
642  seconds += (time_t) sec;
643  return (seconds);
644 }
645 
646 
654 MEMSPACE
655 time_t normalize(tm_t *t, int normalize_to_timezone)
656 {
657  time_t epoch;
658  int32_t offset;
659  int isdst;
660 
661 // struct tm
662 // {
663 // int tm_sec; /*< Seconds. [0-60] (1 leap second) */
664 // int tm_min; /*< Minutes. [0-59] */
665 // int tm_hour; /*< Hours. [0-23] */
666 // int tm_mday; /*< Day. [1-31] */
667 // int tm_mon; /*< Month. [0-11] */
668 // int tm_year; /*< Year - 1900. */
669 // int tm_wday; /*< Day of week. [0-6] */
670 // int tm_yday; /*< Days in year.[0-365] */
671 // int tm_isdst; /*< DST. [-1/0/1] */
672 // };
673 
674 // Normalize t->tm_seconds
675  while(t->tm_sec >= 60)
676  {
677  ++t->tm_min;
678  t->tm_sec -= 60;
679  }
680  while(t->tm_sec < 0)
681  {
682  --t->tm_min;
683  t->tm_sec += 60;
684  }
685 
686 // Normalize t->tm_miniutes
687  while(t->tm_min >= 60)
688  {
689  ++t->tm_hour;
690  t->tm_min -= 60;
691  }
692  while(t->tm_min < 0)
693  {
694  --t->tm_hour;
695  t->tm_min += 60;
696  }
697 
698 // Normalize t->tm_hours
699  while(t->tm_hour >= 24)
700  {
701  ++t->tm_mday;
702  t->tm_hour -= 24;
703  }
704  while(t->tm_hour < 0)
705  {
706  --t->tm_mday;
707  t->tm_hour += 24;
708  }
709 
710 // Normalize t->tm_months
711  while(t->tm_mon >= 12)
712  {
713  ++t->tm_year;
714  t->tm_mon -= 12;
715  }
716  while(t->tm_mon < 0)
717  {
718  --t->tm_year;
719  t->tm_mon += 12;
720  }
721 
722 // Normalize t->tm_mday
723 // t->tm_mday is 1 based
724  while(t->tm_mday > Days_Per_Month(t->tm_mon,t->tm_year) )
725  {
726 // subtract days in current t->tm_month
727  t->tm_mday -= Days_Per_Month(t->tm_mon,t->tm_year);
728 // Keep month normalized
729  if(++t->tm_mon >= 12)
730  {
731  t->tm_mon -= 12;
732  ++t->tm_year;
733  }
734  }
735 
736 // t->tm_mday is 1 based
737  while(t->tm_mday < 1)
738  {
739 // Keep month normalized
740  if(--t->tm_mon < 0)
741  {
742  t->tm_mon += 12;
743  --t->tm_year;
744  }
745 // add days in previous mount
746  t->tm_mday += Days_Per_Month(t->tm_mon,t->tm_year);
747  }
748 
749 // We can now set the remain values by converting to EPOCH and back again
750 // convert to EPOCH based seconds
751  epoch = tm2epoch( t );
752 #ifdef TIME_DEBUG
753  printf("%10lu - epoch-1\n",epoch);
754 #endif
755 // Normaize to local timezone with DST
756  offset = 0L;
757  isdst = 0;
758  if(normalize_to_timezone)
759  {
760  tz_t tz;
761 
762  gettimezone(&tz);
763  offset = tz.tz_minuteswest * 60L;
764  if(is_dst(epoch))
765  {
766  offset -= 3600L;
767  isdst = 1;
768  }
769  }
770 #ifdef TIME_DEBUG
771  printf("%10lu - epoch-2\n",epoch);
772 #endif
773 
774 // convert back to TM structure
775  epoch = time_to_tm(epoch, offset, t);
776  if(isdst)
777  t->tm_isdst = 1;
778 
779 #ifdef TIME_DEBUG
780  printf("%10lu - epoch-3\n",epoch);
781 #endif
782  return( epoch );
783 }
784 
785 
791 MEMSPACE
793 {
796  return(0);
797 }
798 
799 
805 MEMSPACE
807 {
810  return(0);
811 }
812 
813 
821 MEMSPACE
822 int gettimeofday(tv_t *tv, tz_t *tz)
823 {
824  ts_t ts;
825 
826  clock_gettime(0, (ts_t *) &ts);
827 
828  tv->tv_sec = ts.tv_sec;
829  tv->tv_usec = ts.tv_nsec / 1000L;
830 
831  gettimezone(tz);
832  return(0);
833 }
834 
835 
842 MEMSPACE
844 {
845  static ts_t ts;
846  clock_gettime(0, (ts_t *) &ts);
847  if(t != NULL)
848  *t = ts.tv_sec;
849  return(ts.tv_sec);
850 }
851 
852 
860 MEMSPACE
861 int settimeofday(tv_t *tv, tz_t *tz)
862 {
863  ts_t ts;
864 
865  ts.tv_sec = tv->tv_sec;
866  ts.tv_nsec = tv->tv_usec * 1000L;
867 
868  clock_settime(0, (ts_t *) &ts);
869 
870  settimezone(tz);
871 
872  return(0);
873 }
874 
875 
881 MEMSPACE
882 void clock_set(uint32_t seconds, uint32_t us)
883 {
884  ts_t ts;
885  ts.tv_sec = seconds;
886  ts.tv_nsec = us * 1000L;
887 
888 //FIXME we may want to add hooks to the OS time functions if they exist
889  clock_settime(0, (ts_t *) &ts);
890 }
891 
892 
900 MEMSPACE
901 int setdate (void)
902 {
903  char buf[40];
904  extern MEMSPACE char *fgets ( char *str , int size , FILE *stream );
905 
906  printf("Enter date YYYY MM DD HH:MM:SS >");
907  fgets(buf,sizeof(buf)-2,stdin);
908 
909  return(setdate_r(buf));
910 }
911 
912 
919 MEMSPACE
920 int setdate_r (char *buf)
921 {
922  tm_t tm;
923  ts_t ts;
924  time_t seconds;
925 
927 
928 #ifdef SMALL_SSCANF
929  sscanf(buf,"%d %d %d %d:%d:%d",
930  &tm.tm_year,
931  &tm.tm_mon,
932  &tm.tm_mday,
933  &tm.tm_hour,
934  &tm.tm_min,
935  &tm.tm_sec);
936 #else
937 // Year
938  while(*buf == ' ')
939  ++buf;
940  tm.tm_year = strtol(buf,&buf,10);
941 
942 // Month
943  while(*buf == ' ')
944  ++buf;
945  tm.tm_mon = strtol(buf,&buf,10);
946 
947 // Day of Month
948  while(*buf == ' ')
949  ++buf;
950  tm.tm_mday = strtol(buf,&buf,10);
951 
952 // Hour
953  while(*buf == ' ')
954  ++buf;
955  tm.tm_hour = strtol(buf,&buf,10);
956 
957 // Minute
958  if(*buf && (*buf == ' ' || *buf == ':'))
959  ++buf;
960  tm.tm_min = strtol(buf,&buf,10);
961 
962 // Second
963  if(*buf && (*buf == ' ' || *buf == ':'))
964  ++buf;
965  tm.tm_sec = strtol(buf,&buf,10);
966 #endif
967 
968  tm.tm_mon--;
969 
970  if(tm.tm_year < 1970 || tm.tm_year > 2038)
971  {
972  printf("invalid year: %d\n",tm.tm_year);
973  return(-1);
974  }
975  if(tm.tm_year >= 1900)
976  tm.tm_year -= 1900;
977  if(tm.tm_mon < 0 || tm.tm_mon > 11)
978  {
979  printf("invalid mon: %d\n",tm.tm_year);
980  return(-1);
981  }
982  if(tm.tm_mday < 1 || tm.tm_mday > 31)
983  {
984  printf("invalid day: %d\n",tm.tm_mday);
985  return(-1);
986  }
987  if(tm.tm_hour < 0 || tm.tm_hour > 23)
988  {
989  printf("invalid hour: %d\n",tm.tm_hour);
990  return(-1);
991  }
992  if(tm.tm_min < 0 || tm.tm_min > 59)
993  {
994  printf("invalid min: %d\n",tm.tm_min);
995  return(-1);
996  }
997 
998  seconds = timegm(&tm);
999 
1000  ts.tv_sec = seconds;
1001  ts.tv_nsec = 0L;
1002  clock_settime(0, (ts_t *) &ts);
1003 
1004 #ifdef RTC_SUPPORT
1005  if( !rtc_init(1, (time_t) seconds ) )
1006  {
1007  printf("rtc force init failed\n");
1008  return(-1);
1009  }
1010 #endif
1011  return(0);
1012 }
1013 
1014 
1039 MEMSPACE
1040 time_t find_dst(int dst, time_t epoch, int year, int month, int weekno, int dayno, int hour)
1041 {
1042  tm_t t;
1043  tz_t tz;
1044  tv_t tv;
1045  time_t dst_epoch;
1046  int32_t offset;
1047  int days;
1048 
1049 // Get timezone and clock time
1050  gettimeofday(&tv, &tz);
1051 
1052 // Local time offset in seconds
1053 // Get local timezone offset in seconds without DST offset
1054  offset = tz.tz_minuteswest * 60UL;
1055 // Add DST offset if DST end time includes DST offset
1056  if(dst)
1057  offset -= (60UL * 60UL);
1058 
1059  if(year)
1060  {
1061 // Year is specified - use it
1062  t.tm_year = year - 1900; // 0 = 1900
1063  }
1064  else
1065  {
1066 // Otherwise, Calculate year from epoch or GMT time
1067  if(!epoch)
1068  epoch = tv.tv_sec;
1069  (void) time_to_tm(epoch, offset, &t);
1070  }
1071 
1072 // Local time of DST
1073 // We compute seconds for start of month in which DST happens
1074 // Result from timegm() is in GMT
1075  t.tm_mon = month - 1; // 0..11
1076  t.tm_mday = 1; // 1..28,29,30,31
1077  t.tm_hour = hour; // 0 .. 23
1078  t.tm_min = 0; // 0..59
1079  t.tm_sec = 0; // 0..59
1080 
1081 // Adjust tm_t to localtime - we can not use normalize as it calls us
1082 
1083 // local seconds as timegm does not add offset
1084  dst_epoch = timegm(&t);
1085 
1086 // Add offset for localtime
1087  dst_epoch += offset;
1088 
1089 // Convert to back to tm_t structure with localtime offsets applied
1090  (void) time_to_tm(dst_epoch, 0L, &t);
1091 
1092  dayno = dayno;
1093  weekno= weekno;
1094  days= 0;
1095  while( 1 )
1096  {
1097  if( ((t.tm_wday + days) % 7) == dayno)
1098  {
1099  if( (--weekno) == 0)
1100  break;
1101  }
1102  ++days;
1103  dst_epoch += 86400L;
1104  }
1105 
1106 // Return GMT
1107  return(dst_epoch);
1108 }
1109 
1110 
1114 MEMSPACE
1115 void set_dst(time_t epoch)
1116 {
1117  if(epoch == 0)
1118  {
1119  tv_t tv;
1120  tz_t tz;
1121 // Get timezone and clock time
1122  gettimeofday(&tv, &tz);
1123  epoch = tv.tv_sec;
1124  }
1125 
1126 // Cache caclulations so we do not do them more often then once a day
1127  if(dst.epoch >= epoch)
1128  {
1129  if((dst.epoch - epoch) < 86400L)
1130  return;
1131  }
1132  else
1133  {
1134  if((epoch - dst.epoch) < 86400L)
1135  return;
1136  }
1137 
1138  dst.epoch = epoch;
1139 // FIXME think of ways to cache this computation
1140  dst.start = find_dst(0, epoch, 0, 3, 2, 0, 2);
1141  dst.end = find_dst(1, epoch, 0, 11, 1, 0, 2);
1142 }
1143 
1144 
1149 MEMSPACE
1150 int is_dst(time_t epoch)
1151 {
1152  set_dst(epoch);
1153 
1154  if( epoch >= dst.start && epoch <= dst.end)
1155  return(1);
1156  return(0);
1157 }
1158 
1159 
1161 MEMSPACE
1163 {
1164  printf("DST START localtime: %s\n", ctime(&dst.start));
1165  printf("DST END localtime: %s\n", ctime(&dst.end));
1166 }
1167 
1168 
1170 MEMSPACE
1172 {
1173  printf("DST START GMT: %s\n", ctime_gm(&dst.start));
1174  printf("DST END GMT: %s\n", ctime_gm(&dst.end));
1175 
1176 }
1177 
1178 
1183 MEMSPACE
1184 void initialize_clock(int minwest)
1185 {
1186  time_t seconds = 0;
1187  tm_t tc;
1188  ts_t ts;
1189  tz_t tz;
1190 
1191 #ifdef RTC_SUPPORT
1192  if(!rtc_init(0,0L))
1193  {
1194  printf("rtc uninitilized\n");
1195  printf("attempting rtc init\n");
1196  if( !rtc_init(1, (time_t) 0) )
1197  {
1198  printf("rtc force init failed\n");
1199  }
1200  }
1201 
1202  if(rtc_read(&tc))
1203  {
1204  seconds = timegm(&tc);
1205  }
1206  else
1207  {
1208  seconds = 0;
1209  printf("rtc read errorafter init\n");
1210  }
1211 #else
1212  printf("NO RTC\n");
1213  seconds = 0;
1214 #endif // RTC_SUPPORT
1215  if(!seconds)
1216  printf("use setdate command to change time\n");
1217  tz.tz_minuteswest = minwest;
1218  tz.tz_dsttime = 0;
1219  settimezone( &tz );
1220 
1221  ts.tv_sec = seconds;
1222  ts.tv_nsec = 0L;
1223  clock_settime(0, (ts_t *) &ts);
1224 }
1225 
1226 
1233 MEMSPACE
1235 {
1236  time_t seconds;
1237  tm_t tc;
1238  ts_t ts;
1239 
1240 #ifdef RTC_SUPPORT
1241  if(rtc_read(&tc))
1242  {
1243  seconds = timegm(&tc);
1244  printf("rtc seconds: %lu\n",seconds);
1245  printf("rtc time: %s\n",asctime(&tc));
1246 #if RTC_DEBUG
1247  printf("rtc_read:%d, day:%d,mon:%d,hour:%d,min:%d,sec:%d, wday:%d\n",
1248  (int) tc.tm_year + 1900,
1249  (int) tc.tm_mday,
1250  (int) tc.tm_mon,
1251  (int) tc.tm_hour,
1252  (int) tc.tm_min,
1253  (int) tc.tm_sec,
1254  (int) tc.tm_wday);
1255 #endif
1256  }
1257  else
1258  {
1259  printf("RTC read failed\n");
1260  }
1261 #endif // RTC_SUPPORT
1262 
1263  clock_gettime(0, (ts_t *) &ts);
1264  seconds = ts.tv_sec;
1265  printf("clk seconds: %lu\n",seconds);
1266  printf("clk time: %s\n", asctime(gmtime(&seconds)));
1267 }
gettimeofday
MEMSPACE int gettimeofday(tv_t *tv, tz_t *tz)
Get current time struct timeval *tv and struct timezone *tz - POSIX function. We assume a GMT hardwar...
Definition: time.c:822
clock_settime
MEMSPACE int clock_settime(clockid_t clk_id __attribute__((unused)), const struct timespec *ts)
Set system clock using seconds and nonoseconds - POSIX function.
Definition: timer.c:353
tm2epoch
static MEMSPACE time_t tm2epoch(tm_t *t)
Converts tm_t structure as GMT time into GMT epoch since 1900.
Definition: time.c:553
timeval::tv_usec
uint32_t tv_usec
Definition: time.h:68
fatfs.h
__Month
const char * __Month[]
Short Name or each Month in a year.
Definition: time.c:100
timezone
POSIX timezone.
Definition: time.h:75
dst_t
DST structure.
Definition: time.h:101
__WDay
const char * __WDay[]
Short Name of each Day in a week.
Definition: time.c:78
printf
MEMSPACE int printf(const char *format,...)
stdin
#define stdin
define stdin, stdout and stderr
Definition: posix.h:273
tm::tm_min
int tm_min
Definition: time.h:43
EPOCH_DAY
#define EPOCH_DAY
Definition: time.h:29
find_dst
MEMSPACE time_t find_dst(int dst, time_t epoch, int year, int month, int weekno, int dayno, int hour)
Calculate GMT seconds of DST transition given LOCAL time start / end time and DST flag.
Definition: time.c:1040
settimeofday
MEMSPACE int settimeofday(tv_t *tv, tz_t *tz)
Set current time struct timeval *tv and struct timezone *tz - POSIX function. We assume a GMT hardwar...
Definition: time.c:861
Leap_Days_Since_1900
static MEMSPACE int Leap_Days_Since_1900(int year)
Number of leap days since 1900 to the BEGINNING of the year.
Definition: time.c:156
MEMSPACE
#define MEMSPACE
Definition: user_config.h:17
timegm
MEMSPACE time_t timegm(tm_t *t)
Convert tm_t structure as GMT time into GMT seconds since 1900. All calculactions are in GMT regardle...
Definition: time.c:348
tm::tm_wday
int tm_wday
Definition: time.h:48
__days
static const uint16_t __days[]
days in each month.
Definition: time.c:68
fgets
MEMSPACE char * fgets(char *str, int size, FILE *stream)
get a string from stdin See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:432
clock_set
MEMSPACE void clock_set(uint32_t seconds, uint32_t us)
Set system clock with seconds and microseconds.
Definition: time.c:882
gmtime
MEMSPACE tm_t * gmtime(time_t *tp)
Convert epoch GMT time_t *tp into POSIX static tm_t *t.
Definition: time.c:471
strtol
MEMSPACE long strtol(const char *nptr, char **endptr, int base)
Convert ASCII string to number in a given base.
Definition: mathio.c:139
ctime_gm
MEMSPACE char * ctime_gm(time_t *tp)
GMT version of POSIX ctime().
Definition: time.c:441
dst
dst_t dst
DST start and stop in GMT epoch.
Definition: time.c:52
asctime_r
MEMSPACE char * asctime_r(tm_t *t, char *buf)
Convert tm_t *t structure into POSIX asctime() ASCII string *buf.
Definition: time.c:368
tm::tm_sec
int tm_sec
Definition: time.h:42
tm::tm_mon
int tm_mon
Definition: time.h:46
tm::tm_mday
int tm_mday
Definition: time.h:45
settimezone
MEMSPACE int settimezone(tz_t *tz)
Set current timezone with struct timezone *tz - POSIX function.
Definition: time.c:806
tm::tm_hour
int tm_hour
Definition: time.h:44
EPOCH_YEAR
#define EPOCH_YEAR
Definition: time.h:28
ctime_r
MEMSPACE char * ctime_r(time_t *t, char *buf)
Convert local time_t *t epoch time into POSIX asctime() ASCII string *buf.
Definition: time.c:410
timespec::tv_nsec
long tv_nsec
Definition: time.h:88
is_dst
MEMSPACE int is_dst(time_t epoch)
Test GMT epoch time to see if DST applies in a local timezone.
Definition: time.c:1150
initialize_clock
MEMSPACE void initialize_clock(int minwest)
initialize system time - if we have an RTC use it
Definition: time.c:1184
setdate
MEMSPACE int setdate(void)
Set date and time by prompting user.
Definition: time.c:901
sum
MEMSPACE uint16_t sum(char *name)
tm::tm_isdst
int tm_isdst
Definition: time.h:50
print_dst_gmt
MEMSPACE void print_dst_gmt()
print start/stop for DST as GMT for this year
Definition: time.c:1171
__tzone
tz_t __tzone
System Time Zone.
Definition: time.c:49
timespec::tv_sec
time_t tv_sec
Definition: time.h:87
NULL
#define NULL
Definition: user_config.h:85
__file
FILE type structure.
Definition: posix.h:158
Days_Per_Year
static MEMSPACE int Days_Per_Year(int year)
Find number of days in a given year.
Definition: time.c:210
timer.h
timer functions
time.h
Common Linux/POSIX time functions.
tm_mon_to_ascii
MEMSPACE char * tm_mon_to_ascii(int i)
Get string Short name of Month from month number.
Definition: time.c:114
localtime_r
MEMSPACE tm_t * localtime_r(time_t *t, tm_t *result)
Convert POSIX epoch time_t *tp into POSIX tm_t *result expressed as local time using timezone and DST...
Definition: time.c:498
dst_t::end
time_t end
Start of local DST in GMT.
Definition: time.h:104
clock_gettime
MEMSPACE int clock_gettime(clockid_t clk_id __attribute__((unused)), struct timespec *ts)
Generic clock_gettime() function WITHOUT high resolution.
Definition: timer.c:386
dst_t::epoch
time_t epoch
End of local DST in GMT.
Definition: time.h:105
time_to_tm
MEMSPACE time_t time_to_tm(time_t epoch, int32_t offset, tm_t *t)
Converts epoch ( seconds from 1 Jan EPOCH_YEAR UTC), offset seconds, to UNIX tm *t.
Definition: time.c:263
tm_wday_to_ascii
MEMSPACE char * tm_wday_to_ascii(int i)
Get string Short name of day from day number.
Definition: time.c:87
display_clock
MEMSPACE void display_clock()
Display system time and optionally RTC time.
Definition: time.c:1234
rtc_init
uint8_t rtc_init(int force, time_t seconds)
Initialize DS1307 rtc if not initialied - or if forced.
Definition: rtc.c:285
timeval::tv_sec
time_t tv_sec
Definition: time.h:67
time
MEMSPACE time_t time(time_t *t)
Return second from epoch - POSIX function.
Definition: time.c:843
__clock
volatile ts_t __clock
System Clock Time.
Definition: timer.c:38
timezone::tz_dsttime
int tz_dsttime
Definition: time.h:78
ctime
MEMSPACE char * ctime(time_t *tp)
Convert local time_t *t epoch time into POSIX asctime() string buf[].
Definition: time.c:424
Days_Per_Month
MEMSPACE int Days_Per_Month(int month, int year)
days in a month
Definition: time.c:223
dst_t::start
time_t start
Definition: time.h:103
timeval
POSIX timeval.
Definition: time.h:65
__days_sum
static const uint16_t __days_sum[]
accumulated days to the start of a month in a year.
Definition: time.c:59
rtc_read
uint8_t rtc_read(tm_t *t)
Read DS1307 RTC into POSIX struct tm * structure.
Definition: rtc.c:189
print_dst
MEMSPACE void print_dst()
print start/stop for DST as localtime for this year
Definition: time.c:1162
localtime
MEMSPACE tm_t * localtime(time_t *tp)
Convert POSIX epoch time_t *tp into POSIX tm_t *result.
Definition: time.c:522
set_dst
MEMSPACE void set_dst(time_t epoch)
Set DST start and end time for the given epoch year.
Definition: time.c:1115
tm
POSIX struct tm.
Definition: time.h:40
tm::tm_year
int tm_year
Definition: time.h:47
mktime
MEMSPACE time_t mktime(tm_t *t)
convert tm_t structure to time_t local time epoch
Definition: time.c:535
time_t
uint32_t time_t
type of EPOCH result.
Definition: time.h:34
rtc.h
DS1307 RTC Driver AVR8.
sscanf
int sscanf(const char *strp, const char *fmt,...)
tm::tm_yday
int tm_yday
Definition: time.h:49
mathio.h
Math IO functions, and verious conversion code with floating point support.
gettimezone
MEMSPACE int gettimezone(tz_t *tz)
Get current timezone in struct timezone *tz - POSIX function.
Definition: time.c:792
timespec
POSIX timespec.
Definition: time.h:85
timezone::tz_minuteswest
int tz_minuteswest
Definition: time.h:77
posix.h
POSIX wrapper for FatFS.
finddayofweek
MEMSPACE int finddayofweek(int year, int month, int day)
return day of week for givenn day, month, year
Definition: time.c:180
setdate_r
MEMSPACE int setdate_r(char *buf)
Set date and time from string in this format "YYYY MM DD HH:MM:SS".
Definition: time.c:920
IS_Leap
static MEMSPACE int IS_Leap(int year)
Check if a year is a leap year.
Definition: time.c:135
normalize
MEMSPACE time_t normalize(tm_t *t, int normalize_to_timezone)
Normalize POSIX tm_t *t struct and convert to epoch time Note: does not deal with DST - by design.
Definition: time.c:655
snprintf
#define snprintf(s, size, format, args...)
Definition: user_config.h:72
asctime
MEMSPACE char * asctime(tm_t *t)
Convert tm_t *t structure into POSIX asctime() ASCII string.
Definition: time.c:394
gmtime_r
MEMSPACE tm_t * gmtime_r(time_t *t, tm_t *result)
Convert epoch GMT time_t *tp into POSIX tm_t *result.
Definition: time.c:456