Line data Source code
1 : /**********************************************************************
2 : *
3 : * Name: cpl_time.cpp
4 : * Project: CPL - Common Portability Library
5 : * Purpose: Time functions.
6 : * Author: Even Rouault, <even dot rouault at spatialys.com>
7 : *
8 : **********************************************************************
9 : *
10 : * CPLUnixTimeToYMDHMS() is derived from timesub() in localtime.c from
11 : * openbsd/freebsd/netbsd.
12 : *
13 : * CPLYMDHMSToUnixTime() has been implemented by Even Rouault and is in the
14 : * public domain.
15 : *
16 : * c.f.
17 : *http://svn.freebsd.org/viewvc/base/stable/7/lib/libc/stdtime/localtime.c?revision=178142&view=markup
18 : * localtime.c comes with the following header :
19 : *
20 : * This file is in the public domain, so clarified as of
21 : * 1996-06-05 by Arthur David Olson (arthur_david_olson@nih.gov).
22 : */
23 :
24 : #include "cpl_time.h"
25 : #include "cpl_string.h"
26 :
27 : #include <cstring>
28 : #include <ctime>
29 :
30 : #include "cpl_error.h"
31 :
32 : constexpr int SECSPERMIN = 60;
33 : constexpr int MINSPERHOUR = 60;
34 : constexpr int HOURSPERDAY = 24;
35 : constexpr int SECSPERHOUR = SECSPERMIN * MINSPERHOUR;
36 : constexpr int SECSPERDAY = SECSPERHOUR * HOURSPERDAY;
37 : constexpr int DAYSPERWEEK = 7;
38 : constexpr int MONSPERYEAR = 12;
39 :
40 : constexpr int EPOCH_YEAR = 1970;
41 : constexpr int EPOCH_WDAY = 4;
42 : constexpr int TM_YEAR_BASE = 1900;
43 : constexpr int DAYSPERNYEAR = 365;
44 : constexpr int DAYSPERLYEAR = 366;
45 :
46 74239 : static bool isleap(int y)
47 : {
48 74239 : return ((y % 4) == 0 && (y % 100) != 0) || (y % 400) == 0;
49 : }
50 :
51 104738 : static int LEAPS_THROUGH_END_OF(int y)
52 : {
53 104738 : return y / 4 - y / 100 + y / 400;
54 : }
55 :
56 : constexpr int mon_lengths[2][MONSPERYEAR] = {
57 : {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31},
58 : {31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}};
59 :
60 : constexpr int year_lengths[2] = {DAYSPERNYEAR, DAYSPERLYEAR};
61 :
62 : /************************************************************************/
63 : /* CPLUnixTimeToYMDHMS() */
64 : /************************************************************************/
65 :
66 : /** Converts a time value since the Epoch (aka "unix" time) to a broken-down
67 : * UTC time.
68 : *
69 : * This function is similar to gmtime_r().
70 : * This function will always set tm_isdst to 0.
71 : *
72 : * @param unixTime number of seconds since the Epoch.
73 : * @param pRet address of the return structure.
74 : *
75 : * @return the structure pointed by pRet filled with a broken-down UTC time.
76 : */
77 :
78 29512 : struct tm *CPLUnixTimeToYMDHMS(GIntBig unixTime, struct tm *pRet)
79 : {
80 29512 : GIntBig days = unixTime / SECSPERDAY;
81 29512 : GIntBig rem = unixTime % SECSPERDAY;
82 :
83 29512 : constexpr GIntBig TEN_THOUSAND_YEARS =
84 : static_cast<GIntBig>(10000) * SECSPERDAY * DAYSPERLYEAR;
85 29512 : if (unixTime < -TEN_THOUSAND_YEARS || unixTime > TEN_THOUSAND_YEARS)
86 : {
87 0 : CPLError(CE_Failure, CPLE_NotSupported,
88 : "Invalid unixTime = " CPL_FRMT_GIB, unixTime);
89 0 : memset(pRet, 0, sizeof(*pRet));
90 0 : return pRet;
91 : }
92 :
93 29532 : while (rem < 0)
94 : {
95 20 : rem += SECSPERDAY;
96 20 : --days;
97 : }
98 :
99 29512 : pRet->tm_hour = static_cast<int>(rem / SECSPERHOUR);
100 29512 : rem = rem % SECSPERHOUR;
101 29512 : pRet->tm_min = static_cast<int>(rem / SECSPERMIN);
102 : /*
103 : ** A positive leap second requires a special
104 : ** representation. This uses "... ??:59:60" et seq.
105 : */
106 29512 : pRet->tm_sec = static_cast<int>(rem % SECSPERMIN);
107 29512 : pRet->tm_wday = static_cast<int>((EPOCH_WDAY + days) % DAYSPERWEEK);
108 29512 : if (pRet->tm_wday < 0)
109 25 : pRet->tm_wday += DAYSPERWEEK;
110 :
111 29512 : int y = EPOCH_YEAR;
112 29512 : int yleap = 0;
113 29512 : int iters = 0;
114 117912 : while (iters < 1000 &&
115 55135 : (days < 0 ||
116 55135 : days >= static_cast<GIntBig>(year_lengths[yleap = isleap(y)])))
117 : {
118 33265 : int newy = y + static_cast<int>(days / DAYSPERNYEAR);
119 33265 : if (days < 0)
120 7642 : --newy;
121 99795 : days -= static_cast<GIntBig>(newy - y) * DAYSPERNYEAR +
122 33265 : LEAPS_THROUGH_END_OF(newy - 1) - LEAPS_THROUGH_END_OF(y - 1);
123 33265 : y = newy;
124 33265 : iters++;
125 : }
126 29512 : if (iters == 1000)
127 : {
128 0 : CPLError(CE_Failure, CPLE_NotSupported,
129 : "Invalid unixTime = " CPL_FRMT_GIB, unixTime);
130 0 : memset(pRet, 0, sizeof(*pRet));
131 0 : return pRet;
132 : }
133 :
134 29512 : pRet->tm_year = static_cast<int>(y - TM_YEAR_BASE);
135 29512 : pRet->tm_yday = static_cast<int>(days);
136 29512 : const int *ip = mon_lengths[yleap];
137 :
138 115993 : for (pRet->tm_mon = 0; days >= static_cast<GIntBig>(ip[pRet->tm_mon]);
139 86481 : ++(pRet->tm_mon))
140 86481 : days = days - static_cast<GIntBig>(ip[pRet->tm_mon]);
141 :
142 29512 : pRet->tm_mday = static_cast<int>((days + 1));
143 29512 : pRet->tm_isdst = 0;
144 :
145 29512 : return pRet;
146 : }
147 :
148 : /************************************************************************/
149 : /* CPLYMDHMSToUnixTime() */
150 : /************************************************************************/
151 :
152 : /** Converts a broken-down UTC time into time since the Epoch (aka "unix" time).
153 : *
154 : * This function is similar to mktime(), but the passed structure is not
155 : * modified. This function ignores the tm_wday, tm_yday and tm_isdst fields of
156 : * the passed value. No timezone shift will be applied. This function
157 : * returns 0 for the 1/1/1970 00:00:00
158 : *
159 : * @param brokendowntime broken-downtime UTC time.
160 : *
161 : * @return a number of seconds since the Epoch encoded as a value of type
162 : * GIntBig, or -1 if the time cannot be represented.
163 : */
164 :
165 19107 : GIntBig CPLYMDHMSToUnixTime(const struct tm *brokendowntime)
166 : {
167 :
168 19107 : if (brokendowntime->tm_mon < 0 || brokendowntime->tm_mon >= 12)
169 3 : return -1;
170 :
171 : // Number of days of the current month.
172 19104 : GIntBig days = brokendowntime->tm_mday - 1;
173 :
174 : // Add the number of days of the current year.
175 : const int *ip = mon_lengths[static_cast<int>(
176 19104 : isleap(TM_YEAR_BASE + brokendowntime->tm_year))];
177 85637 : for (int mon = 0; mon < brokendowntime->tm_mon; mon++)
178 66533 : days += ip[mon];
179 :
180 : // Add the number of days of the other years.
181 38208 : days += (TM_YEAR_BASE + static_cast<GIntBig>(brokendowntime->tm_year) -
182 19104 : EPOCH_YEAR) *
183 19104 : DAYSPERNYEAR +
184 38208 : LEAPS_THROUGH_END_OF(static_cast<int>(TM_YEAR_BASE) +
185 19104 : static_cast<int>(brokendowntime->tm_year) -
186 19104 : static_cast<int>(1)) -
187 19104 : LEAPS_THROUGH_END_OF(EPOCH_YEAR - 1);
188 :
189 : // Now add the secondes, minutes and hours to the number of days
190 : // since EPOCH.
191 19104 : return brokendowntime->tm_sec + brokendowntime->tm_min * SECSPERMIN +
192 19104 : brokendowntime->tm_hour * SECSPERHOUR + days * SECSPERDAY;
193 : }
194 :
195 : /************************************************************************/
196 : /* OGRParseRFC822DateTime() */
197 : /************************************************************************/
198 :
199 : static const char *const aszWeekDayStr[] = {"Mon", "Tue", "Wed", "Thu",
200 : "Fri", "Sat", "Sun"};
201 :
202 : static const char *const aszMonthStr[] = {"Jan", "Feb", "Mar", "Apr",
203 : "May", "Jun", "Jul", "Aug",
204 : "Sep", "Oct", "Nov", "Dec"};
205 :
206 : /** Parse a RFC822 formatted date-time string.
207 : *
208 : * Such as [Fri,] 28 Dec 2007 05:24[:17] GMT
209 : *
210 : * @param pszRFC822DateTime formatted string.
211 : * @param pnYear pointer to int receiving year (like 1980, 2000, etc...), or
212 : * NULL
213 : * @param pnMonth pointer to int receiving month (between 1 and 12), or NULL
214 : * @param pnDay pointer to int receiving day of month (between 1 and 31), or
215 : * NULL
216 : * @param pnHour pointer to int receiving hour of day (between 0 and 23), or
217 : * NULL
218 : * @param pnMinute pointer to int receiving minute (between 0 and 59), or NULL
219 : * @param pnSecond pointer to int receiving second (between 0 and 60, or -1 if
220 : * unknown), or NULL
221 : * @param pnTZFlag pointer to int receiving time zone flag (0=unknown, 100=GMT,
222 : * 101=GMT+15minute, 99=GMT-15minute), or NULL
223 : * @param pnWeekDay pointer to int receiving day of week (between 1 and 7, or 0
224 : * if invalid/unset), or NULL
225 : * @return TRUE if parsing is successful
226 : *
227 : * @since GDAL 2.3
228 : */
229 79 : int CPLParseRFC822DateTime(const char *pszRFC822DateTime, int *pnYear,
230 : int *pnMonth, int *pnDay, int *pnHour, int *pnMinute,
231 : int *pnSecond, int *pnTZFlag, int *pnWeekDay)
232 : {
233 : // Following
234 : // https://www.w3.org/Protocols/rfc822/#z28 :
235 : // [Fri,] 28 Dec 2007 05:24[:17] GMT
236 : char **papszTokens =
237 79 : CSLTokenizeStringComplex(pszRFC822DateTime, " ,:", TRUE, FALSE);
238 79 : char **papszVal = papszTokens;
239 79 : int nTokens = CSLCount(papszTokens);
240 79 : if (nTokens < 5)
241 : {
242 16 : CSLDestroy(papszTokens);
243 16 : return false;
244 : }
245 :
246 63 : if (pnWeekDay)
247 17 : *pnWeekDay = 0;
248 :
249 63 : if (!((*papszVal)[0] >= '0' && (*papszVal)[0] <= '9'))
250 : {
251 39 : if (pnWeekDay)
252 : {
253 23 : for (size_t i = 0; i < CPL_ARRAYSIZE(aszWeekDayStr); ++i)
254 : {
255 22 : if (EQUAL(*papszVal, aszWeekDayStr[i]))
256 : {
257 3 : *pnWeekDay = static_cast<int>(i + 1);
258 3 : break;
259 : }
260 : }
261 : }
262 :
263 39 : ++papszVal;
264 : }
265 :
266 63 : int day = atoi(*papszVal);
267 63 : if (day <= 0 || day >= 32)
268 : {
269 2 : CSLDestroy(papszTokens);
270 2 : return false;
271 : }
272 61 : if (pnDay)
273 60 : *pnDay = day;
274 61 : ++papszVal;
275 :
276 61 : int month = 0;
277 362 : for (int i = 0; i < 12; ++i)
278 : {
279 361 : if (EQUAL(*papszVal, aszMonthStr[i]))
280 : {
281 60 : month = i + 1;
282 60 : break;
283 : }
284 : }
285 61 : if (month == 0)
286 : {
287 1 : CSLDestroy(papszTokens);
288 1 : return false;
289 : }
290 60 : if (pnMonth)
291 59 : *pnMonth = month;
292 60 : ++papszVal;
293 :
294 60 : int year = atoi(*papszVal);
295 60 : if (year < 100 && year >= 30)
296 0 : year += 1900;
297 60 : else if (year < 30 && year >= 0)
298 0 : year += 2000;
299 60 : if (pnYear)
300 59 : *pnYear = year;
301 60 : ++papszVal;
302 :
303 60 : int hour = atoi(*papszVal);
304 60 : if (hour < 0 || hour >= 24)
305 : {
306 2 : CSLDestroy(papszTokens);
307 2 : return false;
308 : }
309 58 : if (pnHour)
310 57 : *pnHour = hour;
311 58 : ++papszVal;
312 :
313 58 : if (*papszVal == nullptr)
314 : {
315 1 : CSLDestroy(papszTokens);
316 1 : return false;
317 : }
318 57 : int minute = atoi(*papszVal);
319 57 : if (minute < 0 || minute >= 60)
320 : {
321 2 : CSLDestroy(papszTokens);
322 2 : return false;
323 : }
324 55 : if (pnMinute)
325 54 : *pnMinute = minute;
326 55 : ++papszVal;
327 :
328 55 : if (*papszVal != nullptr && (*papszVal)[0] >= '0' && (*papszVal)[0] <= '9')
329 : {
330 53 : int second = atoi(*papszVal);
331 53 : if (second < 0 || second >= 61)
332 : {
333 1 : CSLDestroy(papszTokens);
334 1 : return false;
335 : }
336 52 : if (pnSecond)
337 51 : *pnSecond = second;
338 52 : ++papszVal;
339 : }
340 2 : else if (pnSecond)
341 2 : *pnSecond = -1;
342 :
343 54 : int TZ = 0;
344 54 : if (*papszVal == nullptr)
345 : {
346 : }
347 32 : else if (strlen(*papszVal) == 5 &&
348 25 : ((*papszVal)[0] == '+' || (*papszVal)[0] == '-'))
349 : {
350 25 : char szBuf[3] = {(*papszVal)[1], (*papszVal)[2], 0};
351 25 : const int TZHour = atoi(szBuf);
352 25 : if (TZHour < 0 || TZHour >= 15)
353 : {
354 2 : CSLDestroy(papszTokens);
355 2 : return false;
356 : }
357 23 : szBuf[0] = (*papszVal)[3];
358 23 : szBuf[1] = (*papszVal)[4];
359 23 : szBuf[2] = 0;
360 23 : const int TZMinute = atoi(szBuf);
361 23 : TZ = 100 + (((*papszVal)[0] == '+') ? 1 : -1) *
362 23 : ((TZHour * 60 + TZMinute) / 15);
363 : }
364 : else
365 : {
366 7 : const char *aszTZStr[] = {"GMT", "UT", "Z", "EST", "EDT", "CST",
367 : "CDT", "MST", "MDT", "PST", "PDT"};
368 7 : const int anTZVal[] = {0, 0, 0, -5, -4, -6, -5, -7, -6, -8, -7};
369 7 : TZ = -1;
370 29 : for (int i = 0; i < 11; ++i)
371 : {
372 27 : if (EQUAL(*papszVal, aszTZStr[i]))
373 : {
374 5 : TZ = 100 + anTZVal[i] * 4;
375 5 : break;
376 : }
377 : }
378 7 : if (TZ < 0)
379 : {
380 2 : CSLDestroy(papszTokens);
381 2 : return false;
382 : }
383 : }
384 :
385 50 : if (pnTZFlag)
386 25 : *pnTZFlag = TZ;
387 :
388 50 : CSLDestroy(papszTokens);
389 50 : return true;
390 : }
|