2 /* Parse a string into an internal time stamp.
3 Copyright (C) 1999, 2000, 2002 Free Software Foundation, Inc.
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2, or (at your option)
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software Foundation,
17 Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
19 /* Originally written by Steven M. Bellovin <smb@research.att.com> while
20 at the University of North Carolina at Chapel Hill. Later tweaked by
21 a couple of people on Usenet. Completely overhauled by Rich $alz
22 <rsalz@bbn.com> and Jim Berets <jberets@bbn.com> in August, 1990.
24 Modified by Paul Eggert <eggert@twinsun.com> in August 1999 to do
25 the right thing about local DST. Unlike previous versions, this
26 version is reentrant. */
35 /* Since the code of getdate.y is not included in the Emacs executable
36 itself, there is no need to #define static in this file. Even if
37 the code were included in the Emacs executable, it probably
38 wouldn't do any harm to #undef it here; this will only cause
39 problems if we try to write to a static variable, which I don't
40 think this code needs to do. */
48 # include <stdlib.h> /* for `free'; used by Bison 1.27 */
51 #if STDC_HEADERS || (! defined isascii && ! HAVE_ISASCII)
52 # define IN_CTYPE_DOMAIN(c) 1
54 # define IN_CTYPE_DOMAIN(c) isascii (c)
57 #define ISSPACE(c) (IN_CTYPE_DOMAIN (c) && isspace (c))
58 #define ISALPHA(c) (IN_CTYPE_DOMAIN (c) && isalpha (c))
59 #define ISLOWER(c) (IN_CTYPE_DOMAIN (c) && islower (c))
60 #define ISDIGIT_LOCALE(c) (IN_CTYPE_DOMAIN (c) && isdigit (c))
62 /* ISDIGIT differs from ISDIGIT_LOCALE, as follows:
63 - Its arg may be any int or unsigned int; it need not be an unsigned char.
64 - It's guaranteed to evaluate its argument exactly once.
65 - It's typically faster.
66 POSIX says that only '0' through '9' are digits. Prefer ISDIGIT to
67 ISDIGIT_LOCALE unless it's important to use the locale's definition
68 of `digit' even when the host does not conform to POSIX. */
69 #define ISDIGIT(c) ((unsigned) (c) - '0' <= 9)
71 #if STDC_HEADERS || HAVE_STRING_H
75 #if __GNUC__ < 2 || (__GNUC__ == 2 && __GNUC_MINOR__ < 8) || __STRICT_ANSI__
76 # define __attribute__(x)
79 #ifndef ATTRIBUTE_UNUSED
80 # define ATTRIBUTE_UNUSED __attribute__ ((__unused__))
83 #define EPOCH_YEAR 1970
84 #define TM_YEAR_BASE 1900
86 #define HOUR(x) ((x) * 60)
88 /* An integer value, and the number of digits in its textual
96 /* An entry in the lexical lookup table. */
104 /* Meridian: am, pm, or 24-hour style. */
105 enum { MERam
, MERpm
, MER24
};
107 /* Information passed to and from the parser. */
110 /* The input string remaining to be parsed. */
113 /* N, if this is the Nth Tuesday. */
116 /* Day of week; Sunday is 0. */
119 /* tm_isdst flag for the local zone. */
122 /* Time zone, in minutes east of UTC. */
125 /* Style used for time. */
128 /* Gregorian year, month, day, hour, minutes, and seconds. */
136 /* Relative year, month, day, hour, minutes, and seconds. */
144 /* Counts of nonterminals of various flavors parsed so far. */
147 int local_zones_seen
;
152 /* Table of local time zone abbrevations, terminated by a null entry. */
153 table local_time_zone_table
[3];
156 #define PC (* (parser_control *) parm)
157 #define YYLEX_PARAM parm
158 #define YYPARSE_PARAM parm
160 static int yyerror ();
165 /* We want a reentrant parser. */
168 /* This grammar has 13 shift/reduce conflicts. */
179 %token
<intval
> tDAY tDAY_UNIT tDAYZONE tHOUR_UNIT tLOCAL_ZONE tMERIDIAN
180 %token
<intval
> tMINUTE_UNIT tMONTH tMONTH_UNIT tSEC_UNIT tYEAR_UNIT tZONE
182 %token
<textintval
> tSNUMBER tUNUMBER
184 %type
<intval
> o_merid
197 { PC.local_zones_seen
++; }
217 | tUNUMBER
':' tUNUMBER o_merid
220 PC.minutes
= $3.value
;
224 | tUNUMBER
':' tUNUMBER tSNUMBER
227 PC.minutes
= $3.value
;
230 PC.time_zone
= $4.value %
100 + ($4.value
/ 100) * 60;
232 | tUNUMBER
':' tUNUMBER
':' tUNUMBER o_merid
235 PC.minutes
= $3.value
;
236 PC.seconds
= $5.value
;
239 | tUNUMBER
':' tUNUMBER
':' tUNUMBER tSNUMBER
242 PC.minutes
= $3.value
;
243 PC.seconds
= $5.value
;
246 PC.time_zone
= $6.value %
100 + ($6.value
/ 100) * 60;
252 { PC.local_isdst
= $1; }
254 { PC.local_isdst
= $1 < 0 ?
1 : $1 + 1; }
259 { PC.time_zone
= $1; }
261 { PC.time_zone
= $1 + 60; }
263 { PC.time_zone
= $1 + 60; }
279 PC.day_ordinal
= $1.value
;
285 tUNUMBER
'/' tUNUMBER
290 | tUNUMBER
'/' tUNUMBER
'/' tUNUMBER
292 /* Interpret as YYYY/MM/DD if the first value has 4 or more digits,
293 otherwise as MM/DD/YY.
294 The goal in recognizing YYYY/MM/DD is solely to support legacy
295 machine-generated dates like those in an RCS log listing. If
296 you want portability, use the ISO 8601 format. */
310 | tUNUMBER tSNUMBER tSNUMBER
312 /* ISO 8601 format. YYYY-MM-DD. */
314 PC.month
= -$2.value
;
317 | tUNUMBER tMONTH tSNUMBER
319 /* e.g. 17-JUN-1992. */
322 PC.year.value
= -$3.value
;
323 PC.year.digits
= $3.digits
;
330 | tMONTH tUNUMBER
',' tUNUMBER
341 | tUNUMBER tMONTH tUNUMBER
352 PC.rel_seconds
= -PC.rel_seconds
;
353 PC.rel_minutes
= -PC.rel_minutes
;
354 PC.rel_hour
= -PC.rel_hour
;
355 PC.rel_day
= -PC.rel_day
;
356 PC.rel_month
= -PC.rel_month
;
357 PC.rel_year
= -PC.rel_year
;
364 { PC.rel_year
+= $1.value
* $2; }
365 | tSNUMBER tYEAR_UNIT
366 { PC.rel_year
+= $1.value
* $2; }
368 { PC.rel_year
+= $1; }
369 | tUNUMBER tMONTH_UNIT
370 { PC.rel_month
+= $1.value
* $2; }
371 | tSNUMBER tMONTH_UNIT
372 { PC.rel_month
+= $1.value
* $2; }
374 { PC.rel_month
+= $1; }
376 { PC.rel_day
+= $1.value
* $2; }
378 { PC.rel_day
+= $1.value
* $2; }
380 { PC.rel_day
+= $1; }
381 | tUNUMBER tHOUR_UNIT
382 { PC.rel_hour
+= $1.value
* $2; }
383 | tSNUMBER tHOUR_UNIT
384 { PC.rel_hour
+= $1.value
* $2; }
386 { PC.rel_hour
+= $1; }
387 | tUNUMBER tMINUTE_UNIT
388 { PC.rel_minutes
+= $1.value
* $2; }
389 | tSNUMBER tMINUTE_UNIT
390 { PC.rel_minutes
+= $1.value
* $2; }
392 { PC.rel_minutes
+= $1; }
394 { PC.rel_seconds
+= $1.value
* $2; }
396 { PC.rel_seconds
+= $1.value
* $2; }
398 { PC.rel_seconds
+= $1; }
405 && ! PC.rels_seen
&& (PC.times_seen ||
2 < $1.digits
))
412 PC.day
= $1.value %
100;
413 PC.month
= ($1.value
/ 100) %
100;
414 PC.year.value
= $1.value
/ 10000;
415 PC.year.digits
= $1.digits
- 4;
427 PC.hour
= $1.value
/ 100;
428 PC.minutes
= $1.value %
100;
446 /* Include this file down here because bison inserts code above which
447 may define-away `const'. We want the prototype for get_date to have
448 the same signature as the function definition. */
449 #include "modules/getdate.h"
452 struct tm
*gmtime
();
455 struct tm
*localtime
();
461 static table
const meridian_table
[] =
463 { "AM", tMERIDIAN
, MERam
},
464 { "A.M.", tMERIDIAN
, MERam
},
465 { "PM", tMERIDIAN
, MERpm
},
466 { "P.M.", tMERIDIAN
, MERpm
},
470 static table
const dst_table
[] =
475 static table
const month_and_day_table
[] =
477 { "JANUARY", tMONTH
, 1 },
478 { "FEBRUARY", tMONTH
, 2 },
479 { "MARCH", tMONTH
, 3 },
480 { "APRIL", tMONTH
, 4 },
481 { "MAY", tMONTH
, 5 },
482 { "JUNE", tMONTH
, 6 },
483 { "JULY", tMONTH
, 7 },
484 { "AUGUST", tMONTH
, 8 },
485 { "SEPTEMBER",tMONTH
, 9 },
486 { "SEPT", tMONTH
, 9 },
487 { "OCTOBER", tMONTH
, 10 },
488 { "NOVEMBER", tMONTH
, 11 },
489 { "DECEMBER", tMONTH
, 12 },
490 { "SUNDAY", tDAY
, 0 },
491 { "MONDAY", tDAY
, 1 },
492 { "TUESDAY", tDAY
, 2 },
494 { "WEDNESDAY",tDAY
, 3 },
495 { "WEDNES", tDAY
, 3 },
496 { "THURSDAY", tDAY
, 4 },
498 { "THURS", tDAY
, 4 },
499 { "FRIDAY", tDAY
, 5 },
500 { "SATURDAY", tDAY
, 6 },
504 static table
const time_units_table
[] =
506 { "YEAR", tYEAR_UNIT
, 1 },
507 { "MONTH", tMONTH_UNIT
, 1 },
508 { "FORTNIGHT",tDAY_UNIT
, 14 },
509 { "WEEK", tDAY_UNIT
, 7 },
510 { "DAY", tDAY_UNIT
, 1 },
511 { "HOUR", tHOUR_UNIT
, 1 },
512 { "MINUTE", tMINUTE_UNIT
, 1 },
513 { "MIN", tMINUTE_UNIT
, 1 },
514 { "SECOND", tSEC_UNIT
, 1 },
515 { "SEC", tSEC_UNIT
, 1 },
519 /* Assorted relative-time words. */
520 static table
const relative_time_table
[] =
522 { "TOMORROW", tMINUTE_UNIT
, 24 * 60 },
523 { "YESTERDAY",tMINUTE_UNIT
, - (24 * 60) },
524 { "TODAY", tMINUTE_UNIT
, 0 },
525 { "NOW", tMINUTE_UNIT
, 0 },
526 { "LAST", tUNUMBER
, -1 },
527 { "THIS", tUNUMBER
, 0 },
528 { "NEXT", tUNUMBER
, 1 },
529 { "FIRST", tUNUMBER
, 1 },
530 /*{ "SECOND", tUNUMBER, 2 }, */
531 { "THIRD", tUNUMBER
, 3 },
532 { "FOURTH", tUNUMBER
, 4 },
533 { "FIFTH", tUNUMBER
, 5 },
534 { "SIXTH", tUNUMBER
, 6 },
535 { "SEVENTH", tUNUMBER
, 7 },
536 { "EIGHTH", tUNUMBER
, 8 },
537 { "NINTH", tUNUMBER
, 9 },
538 { "TENTH", tUNUMBER
, 10 },
539 { "ELEVENTH", tUNUMBER
, 11 },
540 { "TWELFTH", tUNUMBER
, 12 },
545 /* The time zone table. This table is necessarily incomplete, as time
546 zone abbreviations are ambiguous; e.g. Australians interpret "EST"
547 as Eastern time in Australia, not as US Eastern Standard Time.
548 You cannot rely on getdate to handle arbitrary time zone
549 abbreviations; use numeric abbreviations like `-0500' instead. */
550 static table
const time_zone_table
[] =
552 { "GMT", tZONE
, HOUR
( 0) }, /* Greenwich Mean */
553 { "UT", tZONE
, HOUR
( 0) }, /* Universal (Coordinated) */
554 { "UTC", tZONE
, HOUR
( 0) },
555 { "WET", tZONE
, HOUR
( 0) }, /* Western European */
556 { "WEST", tDAYZONE
, HOUR
( 0) }, /* Western European Summer */
557 { "BST", tDAYZONE
, HOUR
( 0) }, /* British Summer */
558 { "ART", tZONE
, -HOUR
( 3) }, /* Argentina */
559 { "BRT", tZONE
, -HOUR
( 3) }, /* Brazil */
560 { "BRST", tDAYZONE
, -HOUR
( 3) }, /* Brazil Summer */
561 { "NST", tZONE
, -(HOUR
( 3) + 30) }, /* Newfoundland Standard */
562 { "NDT", tDAYZONE
,-(HOUR
( 3) + 30) }, /* Newfoundland Daylight */
563 { "AST", tZONE
, -HOUR
( 4) }, /* Atlantic Standard */
564 { "ADT", tDAYZONE
, -HOUR
( 4) }, /* Atlantic Daylight */
565 { "CLT", tZONE
, -HOUR
( 4) }, /* Chile */
566 { "CLST", tDAYZONE
, -HOUR
( 4) }, /* Chile Summer */
567 { "EST", tZONE
, -HOUR
( 5) }, /* Eastern Standard */
568 { "EDT", tDAYZONE
, -HOUR
( 5) }, /* Eastern Daylight */
569 { "CST", tZONE
, -HOUR
( 6) }, /* Central Standard */
570 { "CDT", tDAYZONE
, -HOUR
( 6) }, /* Central Daylight */
571 { "MST", tZONE
, -HOUR
( 7) }, /* Mountain Standard */
572 { "MDT", tDAYZONE
, -HOUR
( 7) }, /* Mountain Daylight */
573 { "PST", tZONE
, -HOUR
( 8) }, /* Pacific Standard */
574 { "PDT", tDAYZONE
, -HOUR
( 8) }, /* Pacific Daylight */
575 { "AKST", tZONE
, -HOUR
( 9) }, /* Alaska Standard */
576 { "AKDT", tDAYZONE
, -HOUR
( 9) }, /* Alaska Daylight */
577 { "HST", tZONE
, -HOUR
(10) }, /* Hawaii Standard */
578 { "HAST", tZONE
, -HOUR
(10) }, /* Hawaii-Aleutian Standard */
579 { "HADT", tDAYZONE
, -HOUR
(10) }, /* Hawaii-Aleutian Daylight */
580 { "SST", tZONE
, -HOUR
(12) }, /* Samoa Standard */
581 { "WAT", tZONE
, HOUR
( 1) }, /* West Africa */
582 { "CET", tZONE
, HOUR
( 1) }, /* Central European */
583 { "CEST", tDAYZONE
, HOUR
( 1) }, /* Central European Summer */
584 { "MET", tZONE
, HOUR
( 1) }, /* Middle European */
585 { "MEZ", tZONE
, HOUR
( 1) }, /* Middle European */
586 { "MEST", tDAYZONE
, HOUR
( 1) }, /* Middle European Summer */
587 { "MESZ", tDAYZONE
, HOUR
( 1) }, /* Middle European Summer */
588 { "EET", tZONE
, HOUR
( 2) }, /* Eastern European */
589 { "EEST", tDAYZONE
, HOUR
( 2) }, /* Eastern European Summer */
590 { "CAT", tZONE
, HOUR
( 2) }, /* Central Africa */
591 { "SAST", tZONE
, HOUR
( 2) }, /* South Africa Standard */
592 { "EAT", tZONE
, HOUR
( 3) }, /* East Africa */
593 { "MSK", tZONE
, HOUR
( 3) }, /* Moscow */
594 { "MSD", tDAYZONE
, HOUR
( 3) }, /* Moscow Daylight */
595 { "IST", tZONE
, (HOUR
( 5) + 30) }, /* India Standard */
596 { "SGT", tZONE
, HOUR
( 8) }, /* Singapore */
597 { "KST", tZONE
, HOUR
( 9) }, /* Korea Standard */
598 { "JST", tZONE
, HOUR
( 9) }, /* Japan Standard */
599 { "GST", tZONE
, HOUR
(10) }, /* Guam Standard */
600 { "NZST", tZONE
, HOUR
(12) }, /* New Zealand Standard */
601 { "NZDT", tDAYZONE
, HOUR
(12) }, /* New Zealand Daylight */
605 /* Military time zone table. */
606 static table
const military_table
[] =
608 { "A", tZONE
, -HOUR
( 1) },
609 { "B", tZONE
, -HOUR
( 2) },
610 { "C", tZONE
, -HOUR
( 3) },
611 { "D", tZONE
, -HOUR
( 4) },
612 { "E", tZONE
, -HOUR
( 5) },
613 { "F", tZONE
, -HOUR
( 6) },
614 { "G", tZONE
, -HOUR
( 7) },
615 { "H", tZONE
, -HOUR
( 8) },
616 { "I", tZONE
, -HOUR
( 9) },
617 { "K", tZONE
, -HOUR
(10) },
618 { "L", tZONE
, -HOUR
(11) },
619 { "M", tZONE
, -HOUR
(12) },
620 { "N", tZONE
, HOUR
( 1) },
621 { "O", tZONE
, HOUR
( 2) },
622 { "P", tZONE
, HOUR
( 3) },
623 { "Q", tZONE
, HOUR
( 4) },
624 { "R", tZONE
, HOUR
( 5) },
625 { "S", tZONE
, HOUR
( 6) },
626 { "T", tZONE
, HOUR
( 7) },
627 { "U", tZONE
, HOUR
( 8) },
628 { "V", tZONE
, HOUR
( 9) },
629 { "W", tZONE
, HOUR
(10) },
630 { "X", tZONE
, HOUR
(11) },
631 { "Y", tZONE
, HOUR
(12) },
632 { "Z", tZONE
, HOUR
( 0) },
639 to_hour
(int hours
, int meridian
)
644 return
0 <= hours
&& hours
< 24 ? hours
: -1;
646 return
0 < hours
&& hours
< 12 ? hours
: hours
== 12 ?
0 : -1;
648 return
0 < hours
&& hours
< 12 ? hours
+ 12 : hours
== 12 ?
12 : -1;
657 to_year
(textint textyear
)
659 int year
= textyear.value
;
664 /* XPG4 suggests that years 00-68 map to 2000-2068, and
665 years 69-99 map to 1969-1999. */
666 if
(textyear.digits
== 2)
667 year
+= year
< 69 ?
2000 : 1900;
673 lookup_zone
(parser_control
const *pc
, char const *name
)
677 /* Try local zone abbreviations first; they're more likely to be right. */
678 for
(tp
= pc
->local_time_zone_table
; tp
->name
; tp
++)
679 if
(strcmp
(name
, tp
->name
) == 0)
682 for
(tp
= time_zone_table
; tp
->name
; tp
++)
683 if
(strcmp
(name
, tp
->name
) == 0)
690 /* Yield the difference between *A and *B,
691 measured in seconds, ignoring leap seconds.
692 The body of this function is taken directly from the GNU C Library;
693 see src/strftime.c. */
695 tm_diff
(struct tm
const *a
, struct tm
const *b
)
697 /* Compute intervening leap days correctly even if year is negative.
698 Take care to avoid int overflow in leap day calculations,
699 but it's OK to assume that A and B are close to each other. */
700 int a4
= (a
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (a
->tm_year
& 3);
701 int b4
= (b
->tm_year
>> 2) + (TM_YEAR_BASE
>> 2) - ! (b
->tm_year
& 3);
702 int a100
= a4
/ 25 - (a4 %
25 < 0);
703 int b100
= b4
/ 25 - (b4 %
25 < 0);
704 int a400
= a100
>> 2;
705 int b400
= b100
>> 2;
706 int intervening_leap_days
= (a4
- b4
) - (a100
- b100
) + (a400
- b400
);
707 int years
= a
->tm_year
- b
->tm_year
;
708 int days
= (365 * years
+ intervening_leap_days
709 + (a
->tm_yday
- b
->tm_yday
));
710 return
(60 * (60 * (24 * days
+ (a
->tm_hour
- b
->tm_hour
))
711 + (a
->tm_min
- b
->tm_min
))
712 + (a
->tm_sec
- b
->tm_sec
));
714 #endif /* ! HAVE_TM_GMTOFF */
717 lookup_word
(parser_control
const *pc
, char *word
)
726 /* Make it uppercase. */
727 for
(p
= word
; *p
; p
++)
728 if
(ISLOWER
((unsigned char) *p
))
729 *p
= toupper
((unsigned char) *p
);
731 for
(tp
= meridian_table
; tp
->name
; tp
++)
732 if
(strcmp
(word
, tp
->name
) == 0)
735 /* See if we have an abbreviation for a month. */
736 wordlen
= strlen
(word
);
737 abbrev
= wordlen
== 3 ||
(wordlen
== 4 && word
[3] == '.');
739 for
(tp
= month_and_day_table
; tp
->name
; tp
++)
740 if
((abbrev ? strncmp
(word
, tp
->name
, 3) : strcmp
(word
, tp
->name
)) == 0)
743 if
((tp
= lookup_zone
(pc
, word
)))
746 if
(strcmp
(word
, dst_table
[0].name
) == 0)
749 for
(tp
= time_units_table
; tp
->name
; tp
++)
750 if
(strcmp
(word
, tp
->name
) == 0)
753 /* Strip off any plural and try the units table again. */
754 if
(word
[wordlen
- 1] == 'S')
756 word
[wordlen
- 1] = '\0';
757 for
(tp
= time_units_table
; tp
->name
; tp
++)
758 if
(strcmp
(word
, tp
->name
) == 0)
760 word
[wordlen
- 1] = 'S'; /* For "this" in relative_time_table. */
763 for
(tp
= relative_time_table
; tp
->name
; tp
++)
764 if
(strcmp
(word
, tp
->name
) == 0)
767 /* Military time zones. */
769 for
(tp
= military_table
; tp
->name
; tp
++)
770 if
(word
[0] == tp
->name
[0])
773 /* Drop out any periods and try the time zone table again. */
774 for
(i
= 0, p
= q
= word
; (*p
= *q
); q
++)
779 if
(i
&& (tp
= lookup_zone
(pc
, word
)))
786 yylex (YYSTYPE *lvalp
, parser_control
*pc
)
793 while
(c
= *pc
->input
, ISSPACE
(c
))
796 if
(ISDIGIT
(c
) || c
== '-' || c
== '+')
801 if
(c
== '-' || c
== '+')
803 sign
= c
== '-' ?
-1 : 1;
806 /* skip the '-' sign */
815 value
= 10 * value
+ c
- '0';
819 lvalp
->textintval.value
= sign
< 0 ?
-value
: value
;
820 lvalp
->textintval.digits
= p
- pc
->input
;
822 return sign ? tSNUMBER
: tUNUMBER
;
833 if
(p
< buff
+ sizeof buff
- 1)
837 while
(ISALPHA
(c
) || c
== '.');
840 tp
= lookup_word
(pc
, buff
);
843 lvalp
->intval
= tp
->value
;
864 /* Do nothing if the parser reports an error. */
866 yyerror (char *s ATTRIBUTE_UNUSED
)
871 /* Parse a date/time string P. Return the corresponding time_t value,
872 or (time_t) -1 if there is an error. P can be an incomplete or
873 relative time specification; if so, use *NOW as the basis for the
876 get_date
(const char *p
, const time_t *now
)
878 time_t Start
= now ?
*now
: time
(0);
879 struct tm
*tmp
= localtime
(&Start
);
888 pc.year.value
= tmp
->tm_year
+ TM_YEAR_BASE
;
890 pc.month
= tmp
->tm_mon
+ 1;
891 pc.day
= tmp
->tm_mday
;
892 pc.hour
= tmp
->tm_hour
;
893 pc.minutes
= tmp
->tm_min
;
894 pc.seconds
= tmp
->tm_sec
;
895 tm.tm_isdst
= tmp
->tm_isdst
;
908 pc.local_zones_seen
= 0;
911 #if HAVE_STRUCT_TM_TM_ZONE
912 pc.local_time_zone_table
[0].name
= tmp
->tm_zone
;
913 pc.local_time_zone_table
[0].type
= tLOCAL_ZONE
;
914 pc.local_time_zone_table
[0].value
= tmp
->tm_isdst
;
915 pc.local_time_zone_table
[1].name
= 0;
917 /* Probe the names used in the next three calendar quarters, looking
918 for a tm_isdst different from the one we already have. */
921 for
(quarter
= 1; quarter
<= 3; quarter
++)
923 time_t probe
= Start
+ quarter
* (90 * 24 * 60 * 60);
924 struct tm
*probe_tm
= localtime
(&probe
);
925 if
(probe_tm
&& probe_tm
->tm_zone
926 && probe_tm
->tm_isdst
!= pc.local_time_zone_table
[0].value
)
929 pc.local_time_zone_table
[1].name
= probe_tm
->tm_zone
;
930 pc.local_time_zone_table
[1].type
= tLOCAL_ZONE
;
931 pc.local_time_zone_table
[1].value
= probe_tm
->tm_isdst
;
932 pc.local_time_zone_table
[2].name
= 0;
942 extern
char *tzname
[];
945 for
(i
= 0; i
< 2; i
++)
947 pc.local_time_zone_table
[i
].name
= tzname
[i
];
948 pc.local_time_zone_table
[i
].type
= tLOCAL_ZONE
;
949 pc.local_time_zone_table
[i
].value
= i
;
951 pc.local_time_zone_table
[i
].name
= 0;
954 pc.local_time_zone_table
[0].name
= 0;
958 if
(pc.local_time_zone_table
[0].name
&& pc.local_time_zone_table
[1].name
959 && ! strcmp
(pc.local_time_zone_table
[0].name
,
960 pc.local_time_zone_table
[1].name
))
962 /* This locale uses the same abbrevation for standard and
963 daylight times. So if we see that abbreviation, we don't
964 know whether it's daylight time. */
965 pc.local_time_zone_table
[0].value
= -1;
966 pc.local_time_zone_table
[1].name
= 0;
969 if
(yyparse (&pc
) != 0
970 ||
1 < pc.times_seen ||
1 < pc.dates_seen ||
1 < pc.days_seen
971 ||
1 < (pc.local_zones_seen
+ pc.zones_seen
)
972 ||
(pc.local_zones_seen
&& 1 < pc.local_isdst
))
975 tm.tm_year
= to_year
(pc.year
) - TM_YEAR_BASE
+ pc.rel_year
;
976 tm.tm_mon
= pc.month
- 1 + pc.rel_month
;
977 tm.tm_mday
= pc.day
+ pc.rel_day
;
978 if
(pc.times_seen ||
(pc.rels_seen
&& ! pc.dates_seen
&& ! pc.days_seen
))
980 tm.tm_hour
= to_hour
(pc.hour
, pc.meridian
);
983 tm.tm_min
= pc.minutes
;
984 tm.tm_sec
= pc.seconds
;
988 tm.tm_hour
= tm.tm_min
= tm.tm_sec
= 0;
991 /* Let mktime deduce tm_isdst if we have an absolute time stamp,
992 or if the relative time stamp mentions days, months, or years. */
993 if
(pc.dates_seen | pc.days_seen | pc.times_seen | pc.rel_day
994 | pc.rel_month | pc.rel_year
)
997 /* But if the input explicitly specifies local time with or without
998 DST, give mktime that information. */
999 if
(pc.local_zones_seen
)
1000 tm.tm_isdst
= pc.local_isdst
;
1004 Start
= mktime
(&tm
);
1006 if
(Start
== (time_t) -1)
1009 /* Guard against falsely reporting errors near the time_t boundaries
1010 when parsing times in other time zones. For example, if the min
1011 time_t value is 1970-01-01 00:00:00 UTC and we are 8 hours ahead
1012 of UTC, then the min localtime value is 1970-01-01 08:00:00; if
1013 we apply mktime to 1970-01-01 00:00:00 we will get an error, so
1014 we apply mktime to 1970-01-02 08:00:00 instead and adjust the time
1015 zone by 24 hours to compensate. This algorithm assumes that
1016 there is no DST transition within a day of the time_t boundaries. */
1020 if
(tm.tm_year
<= EPOCH_YEAR
- TM_YEAR_BASE
)
1023 pc.time_zone
+= 24 * 60;
1028 pc.time_zone
-= 24 * 60;
1030 Start
= mktime
(&tm
);
1033 if
(Start
== (time_t) -1)
1037 if
(pc.days_seen
&& ! pc.dates_seen
)
1039 tm.tm_mday
+= ((pc.day_number
- tm.tm_wday
+ 7) %
7
1040 + 7 * (pc.day_ordinal
- (0 < pc.day_ordinal
)));
1042 Start
= mktime
(&tm
);
1043 if
(Start
== (time_t) -1)
1049 int delta
= pc.time_zone
* 60;
1050 #ifdef HAVE_TM_GMTOFF
1051 delta
-= tm.tm_gmtoff
;
1053 struct tm
*gmt
= gmtime
(&Start
);
1056 delta
-= tm_diff
(&tm
, gmt
);
1058 if
((Start
< Start
- delta
) != (delta
< 0))
1059 return
-1; /* time_t overflow */
1063 /* Add relative hours, minutes, and seconds. Ignore leap seconds;
1064 i.e. "+ 10 minutes" means 600 seconds, even if one of them is a
1065 leap second. Typically this is not what the user wants, but it's
1066 too hard to do it the other way, because the time zone indicator
1067 must be applied before relative times, and if mktime is applied
1068 again the time zone will be lost. */
1071 long d1
= 60 * 60 * (long) pc.rel_hour
;
1072 time_t t1
= t0
+ d1
;
1073 long d2
= 60 * (long) pc.rel_minutes
;
1074 time_t t2
= t1
+ d2
;
1075 int d3
= pc.rel_seconds
;
1076 time_t t3
= t2
+ d3
;
1077 if
((d1
/ (60 * 60) ^ pc.rel_hour
)
1078 |
(d2
/ 60 ^ pc.rel_minutes
)
1079 |
((t0
+ d1
< t0
) ^
(d1
< 0))
1080 |
((t1
+ d2
< t1
) ^
(d2
< 0))
1081 |
((t2
+ d3
< t2
) ^
(d3
< 0)))
1094 main
(int ac
, char **av
)
1099 printf
("Enter date, or blank line to exit.\n\t> ");
1102 buff
[BUFSIZ
- 1] = 0;
1103 while
(fgets
(buff
, BUFSIZ
- 1, stdin
) && buff
[0])
1105 d
= get_date
(buff
, 0);
1106 if
(d
== (time_t) -1)
1107 printf
("Bad format - couldn't convert.\n");
1109 printf
("%s", ctime
(&d
));
1115 #endif /* defined TEST */