tclGetDate.y
上传用户:rrhhcc
上传日期:2015-12-11
资源大小:54129k
文件大小:33k
源码类别:

通讯编程

开发平台:

Visual C++

  1. /* 
  2.  * tclGetDate.y --
  3.  *
  4.  * Contains yacc grammar for parsing date and time strings.
  5.  * The output of this file should be the file tclDate.c which
  6.  * is used directly in the Tcl sources.
  7.  *
  8.  * Copyright (c) 1992-1995 Karl Lehenbauer and Mark Diekhans.
  9.  * Copyright (c) 1995-1997 Sun Microsystems, Inc.
  10.  *
  11.  * See the file "license.terms" for information on usage and redistribution
  12.  * of this file, and for a DISCLAIMER OF ALL WARRANTIES.
  13.  *
  14.  * RCS: @(#) $Id: tclGetDate.y,v 1.18.4.2 2005/11/04 20:15:09 kennykb Exp $
  15.  */
  16. %{
  17. /* 
  18.  * tclDate.c --
  19.  *
  20.  * This file is generated from a yacc grammar defined in
  21.  * the file tclGetDate.y.  It should not be edited directly.
  22.  *
  23.  * Copyright (c) 1992-1995 Karl Lehenbauer and Mark Diekhans.
  24.  * Copyright (c) 1995-1997 Sun Microsystems, Inc.
  25.  *
  26.  * See the file "license.terms" for information on usage and redistribution
  27.  * of this file, and for a DISCLAIMER OF ALL WARRANTIES.
  28.  *
  29.  * SCCSID
  30.  */
  31. #include "tclInt.h"
  32. #include "tclPort.h"
  33. #if defined(MAC_TCL) && !defined(TCL_MAC_USE_MSL_EPOCH)
  34. #   define EPOCH           1904
  35. #   define START_OF_TIME   1904
  36. #   define END_OF_TIME     2039
  37. #else
  38. #   define EPOCH           1970
  39. #   define START_OF_TIME   1902
  40. #   define END_OF_TIME     2037
  41. #endif
  42. /*
  43.  * The offset of tm_year of struct tm returned by localtime, gmtime, etc.
  44.  * I don't know how universal this is; K&R II, the NetBSD manpages, and
  45.  * ../compat/strftime.c all agree that tm_year is the year-1900.  However,
  46.  * some systems may have a different value.  This #define should be the
  47.  * same as in ../compat/strftime.c.
  48.  */
  49. #define TM_YEAR_BASE 1900
  50. #define HOUR(x)         ((int) (60 * x))
  51. #define SECSPERDAY      (24L * 60L * 60L)
  52. #define IsLeapYear(x)   ((x % 4 == 0) && (x % 100 != 0 || x % 400 == 0))
  53. /*
  54.  *  An entry in the lexical lookup table.
  55.  */
  56. typedef struct _TABLE {
  57.     char        *name;
  58.     int         type;
  59.     time_t      value;
  60. } TABLE;
  61. /*
  62.  *  Daylight-savings mode:  on, off, or not yet known.
  63.  */
  64. typedef enum _DSTMODE {
  65.     DSTon, DSToff, DSTmaybe
  66. } DSTMODE;
  67. /*
  68.  *  Meridian:  am, pm, or 24-hour style.
  69.  */
  70. typedef enum _MERIDIAN {
  71.     MERam, MERpm, MER24
  72. } MERIDIAN;
  73. /*
  74.  *  Global variables.  We could get rid of most of these by using a good
  75.  *  union as the yacc stack.  (This routine was originally written before
  76.  *  yacc had the %union construct.)  Maybe someday; right now we only use
  77.  *  the %union very rarely.
  78.  */
  79. static char     *yyInput;
  80. static DSTMODE  yyDSTmode;
  81. static time_t   yyDayOrdinal;
  82. static time_t   yyDayNumber;
  83. static time_t   yyMonthOrdinal;
  84. static int      yyHaveDate;
  85. static int      yyHaveDay;
  86. static int      yyHaveOrdinalMonth;
  87. static int      yyHaveRel;
  88. static int      yyHaveTime;
  89. static int      yyHaveZone;
  90. static time_t   yyTimezone;
  91. static time_t   yyDay;
  92. static time_t   yyHour;
  93. static time_t   yyMinutes;
  94. static time_t   yyMonth;
  95. static time_t   yySeconds;
  96. static time_t   yyYear;
  97. static MERIDIAN yyMeridian;
  98. static time_t   yyRelMonth;
  99. static time_t   yyRelDay;
  100. static time_t   yyRelSeconds;
  101. static time_t  *yyRelPointer;
  102. /*
  103.  * Prototypes of internal functions.
  104.  */
  105. static void yyerror _ANSI_ARGS_((char *s));
  106. static time_t ToSeconds _ANSI_ARGS_((time_t Hours, time_t Minutes,
  107.     time_t Seconds, MERIDIAN Meridian));
  108. static int Convert _ANSI_ARGS_((time_t Month, time_t Day, time_t Year,
  109.     time_t Hours, time_t Minutes, time_t Seconds,
  110.     MERIDIAN Meridia, DSTMODE DSTmode, time_t *TimePtr));
  111. static time_t DSTcorrect _ANSI_ARGS_((time_t Start, time_t Future));
  112. static time_t NamedDay _ANSI_ARGS_((time_t Start, time_t DayOrdinal,
  113.     time_t DayNumber));
  114. static time_t   NamedMonth _ANSI_ARGS_((time_t Start, time_t MonthOrdinal,
  115.                     time_t MonthNumber));
  116. static int RelativeMonth _ANSI_ARGS_((time_t Start, time_t RelMonth,
  117.     time_t *TimePtr));
  118. static int RelativeDay _ANSI_ARGS_((time_t Start, time_t RelDay,
  119.     time_t *TimePtr));
  120. static int LookupWord _ANSI_ARGS_((char *buff));
  121. static int yylex _ANSI_ARGS_((void));
  122. int
  123. yyparse _ANSI_ARGS_((void));
  124. %}
  125. %union {
  126.     time_t              Number;
  127.     enum _MERIDIAN      Meridian;
  128. }
  129. %token  tAGO tDAY tDAYZONE tID tMERIDIAN tMINUTE_UNIT tMONTH tMONTH_UNIT
  130. %token  tSTARDATE tSEC_UNIT tSNUMBER tUNUMBER tZONE tEPOCH tDST tISOBASE
  131. %token  tDAY_UNIT tNEXT
  132. %type   <Number>        tDAY tDAYZONE tMINUTE_UNIT tMONTH tMONTH_UNIT tDST
  133. %type   <Number>        tSEC_UNIT tSNUMBER tUNUMBER tZONE tISOBASE tDAY_UNIT
  134. %type   <Number>        unit sign tNEXT tSTARDATE
  135. %type   <Meridian>      tMERIDIAN o_merid
  136. %%
  137. spec    : /* NULL */
  138.         | spec item
  139.         ;
  140. item    : time {
  141.             yyHaveTime++;
  142.         }
  143.         | zone {
  144.             yyHaveZone++;
  145.         }
  146.         | date {
  147.             yyHaveDate++;
  148.         }
  149.         | ordMonth {
  150.             yyHaveOrdinalMonth++;
  151.         }
  152.         | day {
  153.             yyHaveDay++;
  154.         }
  155.         | relspec {
  156.             yyHaveRel++;
  157.         }
  158.         | iso {
  159.     yyHaveTime++;
  160.     yyHaveDate++;
  161. }
  162.         | trek {
  163.     yyHaveTime++;
  164.     yyHaveDate++;
  165.     yyHaveRel++;
  166.         }
  167.         | number
  168.         ;
  169. time    : tUNUMBER tMERIDIAN {
  170.             yyHour = $1;
  171.             yyMinutes = 0;
  172.             yySeconds = 0;
  173.             yyMeridian = $2;
  174.         }
  175.         | tUNUMBER ':' tUNUMBER o_merid {
  176.             yyHour = $1;
  177.             yyMinutes = $3;
  178.             yySeconds = 0;
  179.             yyMeridian = $4;
  180.         }
  181.         | tUNUMBER ':' tUNUMBER '-' tUNUMBER {
  182.             yyHour = $1;
  183.             yyMinutes = $3;
  184.             yyMeridian = MER24;
  185.             yyDSTmode = DSToff;
  186.             yyTimezone = ($5 % 100 + ($5 / 100) * 60);
  187.         }
  188.         | tUNUMBER ':' tUNUMBER ':' tUNUMBER o_merid {
  189.             yyHour = $1;
  190.             yyMinutes = $3;
  191.             yySeconds = $5;
  192.             yyMeridian = $6;
  193.         }
  194.         | tUNUMBER ':' tUNUMBER ':' tUNUMBER '-' tUNUMBER {
  195.             yyHour = $1;
  196.             yyMinutes = $3;
  197.             yySeconds = $5;
  198.             yyMeridian = MER24;
  199.             yyDSTmode = DSToff;
  200.             yyTimezone = ($7 % 100 + ($7 / 100) * 60);
  201.         }
  202.         ;
  203. zone    : tZONE tDST {
  204.             yyTimezone = $1;
  205.             yyDSTmode = DSTon;
  206.         }
  207.         | tZONE {
  208.             yyTimezone = $1;
  209.             yyDSTmode = DSToff;
  210.         }
  211.         | tDAYZONE {
  212.             yyTimezone = $1;
  213.             yyDSTmode = DSTon;
  214.         }
  215.         ;
  216. day     : tDAY {
  217.             yyDayOrdinal = 1;
  218.             yyDayNumber = $1;
  219.         }
  220.         | tDAY ',' {
  221.             yyDayOrdinal = 1;
  222.             yyDayNumber = $1;
  223.         }
  224.         | tUNUMBER tDAY {
  225.             yyDayOrdinal = $1;
  226.             yyDayNumber = $2;
  227.         }
  228.         | sign tUNUMBER tDAY {
  229.             yyDayOrdinal = $1 * $2;
  230.             yyDayNumber = $3;
  231.         }
  232.         | tNEXT tDAY {
  233.             yyDayOrdinal = 2;
  234.             yyDayNumber = $2;
  235.         }
  236.         ;
  237. date    : tUNUMBER '/' tUNUMBER {
  238.             yyMonth = $1;
  239.             yyDay = $3;
  240.         }
  241.         | tUNUMBER '/' tUNUMBER '/' tUNUMBER {
  242.             yyMonth = $1;
  243.             yyDay = $3;
  244.             yyYear = $5;
  245.         }
  246.         | tISOBASE {
  247.     yyYear = $1 / 10000;
  248.     yyMonth = ($1 % 10000)/100;
  249.     yyDay = $1 % 100;
  250. }
  251.         | tUNUMBER '-' tMONTH '-' tUNUMBER {
  252.     yyDay = $1;
  253.     yyMonth = $3;
  254.     yyYear = $5;
  255. }
  256.         | tUNUMBER '-' tUNUMBER '-' tUNUMBER {
  257.             yyMonth = $3;
  258.             yyDay = $5;
  259.             yyYear = $1;
  260.         }
  261.         | tMONTH tUNUMBER {
  262.             yyMonth = $1;
  263.             yyDay = $2;
  264.         }
  265.         | tMONTH tUNUMBER ',' tUNUMBER {
  266.             yyMonth = $1;
  267.             yyDay = $2;
  268.             yyYear = $4;
  269.         }
  270.         | tUNUMBER tMONTH {
  271.             yyMonth = $2;
  272.             yyDay = $1;
  273.         }
  274.         | tEPOCH {
  275.     yyMonth = 1;
  276.     yyDay = 1;
  277.     yyYear = EPOCH;
  278. }
  279.         | tUNUMBER tMONTH tUNUMBER {
  280.             yyMonth = $2;
  281.             yyDay = $1;
  282.             yyYear = $3;
  283.         }
  284.         ;
  285. ordMonth: tNEXT tMONTH {
  286.     yyMonthOrdinal = 1;
  287.     yyMonth = $2;
  288. }
  289.         | tNEXT tUNUMBER tMONTH {
  290.     yyMonthOrdinal = $2;
  291.     yyMonth = $3;
  292. }
  293.         ;
  294. iso     : tISOBASE tZONE tISOBASE {
  295.             if ($2 != HOUR(- 7)) YYABORT;
  296.     yyYear = $1 / 10000;
  297.     yyMonth = ($1 % 10000)/100;
  298.     yyDay = $1 % 100;
  299.     yyHour = $3 / 10000;
  300.     yyMinutes = ($3 % 10000)/100;
  301.     yySeconds = $3 % 100;
  302.         }
  303.         | tISOBASE tZONE tUNUMBER ':' tUNUMBER ':' tUNUMBER {
  304.             if ($2 != HOUR(- 7)) YYABORT;
  305.     yyYear = $1 / 10000;
  306.     yyMonth = ($1 % 10000)/100;
  307.     yyDay = $1 % 100;
  308.     yyHour = $3;
  309.     yyMinutes = $5;
  310.     yySeconds = $7;
  311.         }
  312. | tISOBASE tISOBASE {
  313.     yyYear = $1 / 10000;
  314.     yyMonth = ($1 % 10000)/100;
  315.     yyDay = $1 % 100;
  316.     yyHour = $2 / 10000;
  317.     yyMinutes = ($2 % 10000)/100;
  318.     yySeconds = $2 % 100;
  319.         }
  320.         ;
  321. trek    : tSTARDATE tUNUMBER '.' tUNUMBER {
  322.             /*
  323.      * Offset computed year by -377 so that the returned years will
  324.      * be in a range accessible with a 32 bit clock seconds value
  325.      */
  326.             yyYear = $2/1000 + 2323 - 377;
  327.             yyDay  = 1;
  328.     yyMonth = 1;
  329.     yyRelDay += (($2%1000)*(365 + IsLeapYear(yyYear)))/1000;
  330.     yyRelSeconds += $4 * 144 * 60;
  331.         }
  332.         ;
  333. relspec : relunits tAGO {
  334.     yyRelSeconds *= -1;
  335.     yyRelMonth *= -1;
  336.     yyRelDay *= -1;
  337. }
  338. | relunits
  339. ;
  340. relunits : sign tUNUMBER unit  { *yyRelPointer += $1 * $2 * $3; }
  341.         | tUNUMBER unit        { *yyRelPointer += $1 * $2; }
  342.         | tNEXT unit           { *yyRelPointer += $2; }
  343.         | tNEXT tUNUMBER unit  { *yyRelPointer += $2 * $3; }
  344.         | unit                 { *yyRelPointer += $1; }
  345.         ;
  346. sign    : '-'            { $$ = -1; }
  347.         | '+'            { $$ =  1; }
  348.         ;
  349. unit    : tSEC_UNIT      { $$ = $1; yyRelPointer = &yyRelSeconds; }
  350.         | tDAY_UNIT      { $$ = $1; yyRelPointer = &yyRelDay; }
  351.         | tMONTH_UNIT    { $$ = $1; yyRelPointer = &yyRelMonth; }
  352.         ;
  353. number  : tUNUMBER
  354.     {
  355. if (yyHaveTime && yyHaveDate && !yyHaveRel) {
  356.     yyYear = $1;
  357. } else {
  358.     yyHaveTime++;
  359.     if ($1 < 100) {
  360. yyHour = $1;
  361. yyMinutes = 0;
  362.     } else {
  363. yyHour = $1 / 100;
  364. yyMinutes = $1 % 100;
  365.     }
  366.     yySeconds = 0;
  367.     yyMeridian = MER24;
  368. }
  369.     }
  370. ;
  371. o_merid : /* NULL */ {
  372.             $$ = MER24;
  373.         }
  374.         | tMERIDIAN {
  375.             $$ = $1;
  376.         }
  377.         ;
  378. %%
  379. /*
  380.  * Month and day table.
  381.  */
  382. static TABLE    MonthDayTable[] = {
  383.     { "january",        tMONTH,  1 },
  384.     { "february",       tMONTH,  2 },
  385.     { "march",          tMONTH,  3 },
  386.     { "april",          tMONTH,  4 },
  387.     { "may",            tMONTH,  5 },
  388.     { "june",           tMONTH,  6 },
  389.     { "july",           tMONTH,  7 },
  390.     { "august",         tMONTH,  8 },
  391.     { "september",      tMONTH,  9 },
  392.     { "sept",           tMONTH,  9 },
  393.     { "october",        tMONTH, 10 },
  394.     { "november",       tMONTH, 11 },
  395.     { "december",       tMONTH, 12 },
  396.     { "sunday",         tDAY, 0 },
  397.     { "monday",         tDAY, 1 },
  398.     { "tuesday",        tDAY, 2 },
  399.     { "tues",           tDAY, 2 },
  400.     { "wednesday",      tDAY, 3 },
  401.     { "wednes",         tDAY, 3 },
  402.     { "thursday",       tDAY, 4 },
  403.     { "thur",           tDAY, 4 },
  404.     { "thurs",          tDAY, 4 },
  405.     { "friday",         tDAY, 5 },
  406.     { "saturday",       tDAY, 6 },
  407.     { NULL }
  408. };
  409. /*
  410.  * Time units table.
  411.  */
  412. static TABLE    UnitsTable[] = {
  413.     { "year",           tMONTH_UNIT,    12 },
  414.     { "month",          tMONTH_UNIT,     1 },
  415.     { "fortnight",      tDAY_UNIT,      14 },
  416.     { "week",           tDAY_UNIT,       7 },
  417.     { "day",            tDAY_UNIT,       1 },
  418.     { "hour",           tSEC_UNIT, 60 * 60 },
  419.     { "minute",         tSEC_UNIT,      60 },
  420.     { "min",            tSEC_UNIT,      60 },
  421.     { "second",         tSEC_UNIT,       1 },
  422.     { "sec",            tSEC_UNIT,       1 },
  423.     { NULL }
  424. };
  425. /*
  426.  * Assorted relative-time words.
  427.  */
  428. static TABLE    OtherTable[] = {
  429.     { "tomorrow",       tDAY_UNIT,      1 },
  430.     { "yesterday",      tDAY_UNIT,     -1 },
  431.     { "today",          tDAY_UNIT,      0 },
  432.     { "now",            tSEC_UNIT,      0 },
  433.     { "last",           tUNUMBER,      -1 },
  434.     { "this",           tSEC_UNIT,      0 },
  435.     { "next",           tNEXT,          1 },
  436. #if 0
  437.     { "first",          tUNUMBER,       1 },
  438.     { "second",         tUNUMBER,       2 },
  439.     { "third",          tUNUMBER,       3 },
  440.     { "fourth",         tUNUMBER,       4 },
  441.     { "fifth",          tUNUMBER,       5 },
  442.     { "sixth",          tUNUMBER,       6 },
  443.     { "seventh",        tUNUMBER,       7 },
  444.     { "eighth",         tUNUMBER,       8 },
  445.     { "ninth",          tUNUMBER,       9 },
  446.     { "tenth",          tUNUMBER,       10 },
  447.     { "eleventh",       tUNUMBER,       11 },
  448.     { "twelfth",        tUNUMBER,       12 },
  449. #endif
  450.     { "ago",            tAGO,   1 },
  451.     { "epoch",          tEPOCH,   0 },
  452.     { "stardate",       tSTARDATE, 0},
  453.     { NULL }
  454. };
  455. /*
  456.  * The timezone table.  (Note: This table was modified to not use any floating
  457.  * point constants to work around an SGI compiler bug).
  458.  */
  459. static TABLE    TimezoneTable[] = {
  460.     { "gmt",    tZONE,     HOUR( 0) },      /* Greenwich Mean */
  461.     { "ut",     tZONE,     HOUR( 0) },      /* Universal (Coordinated) */
  462.     { "utc",    tZONE,     HOUR( 0) },
  463.     { "uct",    tZONE,     HOUR( 0) },      /* Universal Coordinated Time */
  464.     { "wet",    tZONE,     HOUR( 0) },      /* Western European */
  465.     { "bst",    tDAYZONE,  HOUR( 0) },      /* British Summer */
  466.     { "wat",    tZONE,     HOUR( 1) },      /* West Africa */
  467.     { "at",     tZONE,     HOUR( 2) },      /* Azores */
  468. #if     0
  469.     /* For completeness.  BST is also British Summer, and GST is
  470.      * also Guam Standard. */
  471.     { "bst",    tZONE,     HOUR( 3) },      /* Brazil Standard */
  472.     { "gst",    tZONE,     HOUR( 3) },      /* Greenland Standard */
  473. #endif
  474.     { "nft",    tZONE,     HOUR( 7/2) },    /* Newfoundland */
  475.     { "nst",    tZONE,     HOUR( 7/2) },    /* Newfoundland Standard */
  476.     { "ndt",    tDAYZONE,  HOUR( 7/2) },    /* Newfoundland Daylight */
  477.     { "ast",    tZONE,     HOUR( 4) },      /* Atlantic Standard */
  478.     { "adt",    tDAYZONE,  HOUR( 4) },      /* Atlantic Daylight */
  479.     { "est",    tZONE,     HOUR( 5) },      /* Eastern Standard */
  480.     { "edt",    tDAYZONE,  HOUR( 5) },      /* Eastern Daylight */
  481.     { "cst",    tZONE,     HOUR( 6) },      /* Central Standard */
  482.     { "cdt",    tDAYZONE,  HOUR( 6) },      /* Central Daylight */
  483.     { "mst",    tZONE,     HOUR( 7) },      /* Mountain Standard */
  484.     { "mdt",    tDAYZONE,  HOUR( 7) },      /* Mountain Daylight */
  485.     { "pst",    tZONE,     HOUR( 8) },      /* Pacific Standard */
  486.     { "pdt",    tDAYZONE,  HOUR( 8) },      /* Pacific Daylight */
  487.     { "yst",    tZONE,     HOUR( 9) },      /* Yukon Standard */
  488.     { "ydt",    tDAYZONE,  HOUR( 9) },      /* Yukon Daylight */
  489.     { "hst",    tZONE,     HOUR(10) },      /* Hawaii Standard */
  490.     { "hdt",    tDAYZONE,  HOUR(10) },      /* Hawaii Daylight */
  491.     { "cat",    tZONE,     HOUR(10) },      /* Central Alaska */
  492.     { "ahst",   tZONE,     HOUR(10) },      /* Alaska-Hawaii Standard */
  493.     { "nt",     tZONE,     HOUR(11) },      /* Nome */
  494.     { "idlw",   tZONE,     HOUR(12) },      /* International Date Line West */
  495.     { "cet",    tZONE,    -HOUR( 1) },      /* Central European */
  496.     { "cest",   tDAYZONE, -HOUR( 1) },      /* Central European Summer */
  497.     { "met",    tZONE,    -HOUR( 1) },      /* Middle European */
  498.     { "mewt",   tZONE,    -HOUR( 1) },      /* Middle European Winter */
  499.     { "mest",   tDAYZONE, -HOUR( 1) },      /* Middle European Summer */
  500.     { "swt",    tZONE,    -HOUR( 1) },      /* Swedish Winter */
  501.     { "sst",    tDAYZONE, -HOUR( 1) },      /* Swedish Summer */
  502.     { "fwt",    tZONE,    -HOUR( 1) },      /* French Winter */
  503.     { "fst",    tDAYZONE, -HOUR( 1) },      /* French Summer */
  504.     { "eet",    tZONE,    -HOUR( 2) },      /* Eastern Europe, USSR Zone 1 */
  505.     { "bt",     tZONE,    -HOUR( 3) },      /* Baghdad, USSR Zone 2 */
  506.     { "it",     tZONE,    -HOUR( 7/2) },    /* Iran */
  507.     { "zp4",    tZONE,    -HOUR( 4) },      /* USSR Zone 3 */
  508.     { "zp5",    tZONE,    -HOUR( 5) },      /* USSR Zone 4 */
  509.     { "ist",    tZONE,    -HOUR(11/2) },    /* Indian Standard */
  510.     { "zp6",    tZONE,    -HOUR( 6) },      /* USSR Zone 5 */
  511. #if     0
  512.     /* For completeness.  NST is also Newfoundland Stanard, nad SST is
  513.      * also Swedish Summer. */
  514.     { "nst",    tZONE,    -HOUR(13/2) },    /* North Sumatra */
  515.     { "sst",    tZONE,    -HOUR( 7) },      /* South Sumatra, USSR Zone 6 */
  516. #endif  /* 0 */
  517.     { "wast",   tZONE,    -HOUR( 7) },      /* West Australian Standard */
  518.     { "wadt",   tDAYZONE, -HOUR( 7) },      /* West Australian Daylight */
  519.     { "jt",     tZONE,    -HOUR(15/2) },    /* Java (3pm in Cronusland!) */
  520.     { "cct",    tZONE,    -HOUR( 8) },      /* China Coast, USSR Zone 7 */
  521.     { "jst",    tZONE,    -HOUR( 9) },      /* Japan Standard, USSR Zone 8 */
  522.     { "jdt",    tDAYZONE, -HOUR( 9) },      /* Japan Daylight */
  523.     { "kst",    tZONE,    -HOUR( 9) },      /* Korea Standard */
  524.     { "kdt",    tDAYZONE, -HOUR( 9) },      /* Korea Daylight */
  525.     { "cast",   tZONE,    -HOUR(19/2) },    /* Central Australian Standard */
  526.     { "cadt",   tDAYZONE, -HOUR(19/2) },    /* Central Australian Daylight */
  527.     { "east",   tZONE,    -HOUR(10) },      /* Eastern Australian Standard */
  528.     { "eadt",   tDAYZONE, -HOUR(10) },      /* Eastern Australian Daylight */
  529.     { "gst",    tZONE,    -HOUR(10) },      /* Guam Standard, USSR Zone 9 */
  530.     { "nzt",    tZONE,    -HOUR(12) },      /* New Zealand */
  531.     { "nzst",   tZONE,    -HOUR(12) },      /* New Zealand Standard */
  532.     { "nzdt",   tDAYZONE, -HOUR(12) },      /* New Zealand Daylight */
  533.     { "idle",   tZONE,    -HOUR(12) },      /* International Date Line East */
  534.     /* ADDED BY Marco Nijdam */
  535.     { "dst",    tDST,     HOUR( 0) },       /* DST on (hour is ignored) */
  536.     /* End ADDED */
  537.     {  NULL  }
  538. };
  539. /*
  540.  * Military timezone table.
  541.  */
  542. static TABLE    MilitaryTable[] = {
  543.     { "a",      tZONE,  HOUR(  1) },
  544.     { "b",      tZONE,  HOUR(  2) },
  545.     { "c",      tZONE,  HOUR(  3) },
  546.     { "d",      tZONE,  HOUR(  4) },
  547.     { "e",      tZONE,  HOUR(  5) },
  548.     { "f",      tZONE,  HOUR(  6) },
  549.     { "g",      tZONE,  HOUR(  7) },
  550.     { "h",      tZONE,  HOUR(  8) },
  551.     { "i",      tZONE,  HOUR(  9) },
  552.     { "k",      tZONE,  HOUR( 10) },
  553.     { "l",      tZONE,  HOUR( 11) },
  554.     { "m",      tZONE,  HOUR( 12) },
  555.     { "n",      tZONE,  HOUR(- 1) },
  556.     { "o",      tZONE,  HOUR(- 2) },
  557.     { "p",      tZONE,  HOUR(- 3) },
  558.     { "q",      tZONE,  HOUR(- 4) },
  559.     { "r",      tZONE,  HOUR(- 5) },
  560.     { "s",      tZONE,  HOUR(- 6) },
  561.     { "t",      tZONE,  HOUR(- 7) },
  562.     { "u",      tZONE,  HOUR(- 8) },
  563.     { "v",      tZONE,  HOUR(- 9) },
  564.     { "w",      tZONE,  HOUR(-10) },
  565.     { "x",      tZONE,  HOUR(-11) },
  566.     { "y",      tZONE,  HOUR(-12) },
  567.     { "z",      tZONE,  HOUR(  0) },
  568.     { NULL }
  569. };
  570. /*
  571.  * Dump error messages in the bit bucket.
  572.  */
  573. static void
  574. yyerror(s)
  575.     char  *s;
  576. {
  577. }
  578. static time_t
  579. ToSeconds(Hours, Minutes, Seconds, Meridian)
  580.     time_t      Hours;
  581.     time_t      Minutes;
  582.     time_t      Seconds;
  583.     MERIDIAN    Meridian;
  584. {
  585.     if (Minutes < 0 || Minutes > 59 || Seconds < 0 || Seconds > 59)
  586.         return -1;
  587.     switch (Meridian) {
  588.     case MER24:
  589.         if (Hours < 0 || Hours > 23)
  590.             return -1;
  591.         return (Hours * 60L + Minutes) * 60L + Seconds;
  592.     case MERam:
  593.         if (Hours < 1 || Hours > 12)
  594.             return -1;
  595.         return ((Hours % 12) * 60L + Minutes) * 60L + Seconds;
  596.     case MERpm:
  597.         if (Hours < 1 || Hours > 12)
  598.             return -1;
  599.         return (((Hours % 12) + 12) * 60L + Minutes) * 60L + Seconds;
  600.     }
  601.     return -1;  /* Should never be reached */
  602. }
  603. /*
  604.  *-----------------------------------------------------------------------------
  605.  *
  606.  * Convert --
  607.  *
  608.  *      Convert a {month, day, year, hours, minutes, seconds, meridian, dst}
  609.  *      tuple into a clock seconds value.
  610.  *
  611.  * Results:
  612.  *      0 or -1 indicating success or failure.
  613.  *
  614.  * Side effects:
  615.  *      Fills TimePtr with the computed value.
  616.  *
  617.  *-----------------------------------------------------------------------------
  618.  */
  619. static int
  620. Convert(Month, Day, Year, Hours, Minutes, Seconds, Meridian, DSTmode, TimePtr)
  621.     time_t      Month;
  622.     time_t      Day;
  623.     time_t      Year;
  624.     time_t      Hours;
  625.     time_t      Minutes;
  626.     time_t      Seconds;
  627.     MERIDIAN    Meridian;
  628.     DSTMODE     DSTmode;
  629.     time_t     *TimePtr;
  630. {
  631.     static int  DaysInMonth[12] = {
  632.         31, 0, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31
  633.     };
  634.     time_t tod;
  635.     time_t Julian;
  636.     int i;
  637.     /* Figure out how many days are in February for the given year.
  638.      * Every year divisible by 4 is a leap year.
  639.      * But, every year divisible by 100 is not a leap year.
  640.      * But, every year divisible by 400 is a leap year after all.
  641.      */
  642.     DaysInMonth[1] = IsLeapYear(Year) ? 29 : 28;
  643.     /* Check the inputs for validity */
  644.     if (Month < 1 || Month > 12
  645.     || Year < START_OF_TIME || Year > END_OF_TIME
  646.     || Day < 1 || Day > DaysInMonth[(int)--Month])
  647.         return -1;
  648.     /* Start computing the value.  First determine the number of days
  649.      * represented by the date, then multiply by the number of seconds/day.
  650.      */
  651.     for (Julian = Day - 1, i = 0; i < Month; i++)
  652.         Julian += DaysInMonth[i];
  653.     if (Year >= EPOCH) {
  654.         for (i = EPOCH; i < Year; i++)
  655.             Julian += 365 + IsLeapYear(i);
  656.     } else {
  657.         for (i = Year; i < EPOCH; i++)
  658.             Julian -= 365 + IsLeapYear(i);
  659.     }
  660.     Julian *= SECSPERDAY;
  661.     /* Add the timezone offset ?? */
  662.     Julian += yyTimezone * 60L;
  663.     /* Add the number of seconds represented by the time component */
  664.     if ((tod = ToSeconds(Hours, Minutes, Seconds, Meridian)) < 0)
  665.         return -1;
  666.     Julian += tod;
  667.     /* Perform a preliminary DST compensation ?? */
  668.     if (DSTmode == DSTon
  669.      || (DSTmode == DSTmaybe && TclpGetDate((TclpTime_t)&Julian, 0)->tm_isdst))
  670.         Julian -= 60 * 60;
  671.     *TimePtr = Julian;
  672.     return 0;
  673. }
  674. static time_t
  675. DSTcorrect(Start, Future)
  676.     time_t      Start;
  677.     time_t      Future;
  678. {
  679.     time_t      StartDay;
  680.     time_t      FutureDay;
  681.     StartDay = (TclpGetDate((TclpTime_t)&Start, 0)->tm_hour + 1) % 24;
  682.     FutureDay = (TclpGetDate((TclpTime_t)&Future, 0)->tm_hour + 1) % 24;
  683.     return (Future - Start) + (StartDay - FutureDay) * 60L * 60L;
  684. }
  685. static time_t
  686. NamedDay(Start, DayOrdinal, DayNumber)
  687.     time_t      Start;
  688.     time_t      DayOrdinal;
  689.     time_t      DayNumber;
  690. {
  691.     struct tm   *tm;
  692.     time_t      now;
  693.     now = Start;
  694.     tm = TclpGetDate((TclpTime_t)&now, 0);
  695.     now += SECSPERDAY * ((DayNumber - tm->tm_wday + 7) % 7);
  696.     now += 7 * SECSPERDAY * (DayOrdinal <= 0 ? DayOrdinal : DayOrdinal - 1);
  697.     return DSTcorrect(Start, now);
  698. }
  699. static time_t
  700. NamedMonth(Start, MonthOrdinal, MonthNumber)
  701.     time_t Start;
  702.     time_t MonthOrdinal;
  703.     time_t MonthNumber;
  704. {
  705.     struct tm *tm;
  706.     time_t now;
  707.     int result;
  708.     
  709.     now = Start;
  710.     tm = TclpGetDate((TclpTime_t)&now, 0);
  711.     /* To compute the next n'th month, we use this alg:
  712.      * add n to year value
  713.      * if currentMonth < requestedMonth decrement year value by 1 (so that
  714.      *  doing next february from january gives us february of the current year)
  715.      * set day to 1, time to 0
  716.      */
  717.     tm->tm_year += MonthOrdinal;
  718.     if (tm->tm_mon < MonthNumber - 1) {
  719. tm->tm_year--;
  720.     }
  721.     result = Convert(MonthNumber, (time_t) 1, tm->tm_year + TM_YEAR_BASE,
  722.     (time_t) 0, (time_t) 0, (time_t) 0, MER24, DSTmaybe, &now);
  723.     if (result < 0) {
  724. return 0;
  725.     }
  726.     return DSTcorrect(Start, now);
  727. }
  728. static int
  729. RelativeMonth(Start, RelMonth, TimePtr)
  730.     time_t Start;
  731.     time_t RelMonth;
  732.     time_t *TimePtr;
  733. {
  734.     struct tm *tm;
  735.     time_t Month;
  736.     time_t Year;
  737.     time_t Julian;
  738.     int result;
  739.     if (RelMonth == 0) {
  740.         *TimePtr = 0;
  741.         return 0;
  742.     }
  743.     tm = TclpGetDate((TclpTime_t)&Start, 0);
  744.     Month = 12 * (tm->tm_year + TM_YEAR_BASE) + tm->tm_mon + RelMonth;
  745.     Year = Month / 12;
  746.     Month = Month % 12 + 1;
  747.     result = Convert(Month, (time_t) tm->tm_mday, Year,
  748.     (time_t) tm->tm_hour, (time_t) tm->tm_min, (time_t) tm->tm_sec,
  749.     MER24, DSTmaybe, &Julian);
  750.     /*
  751.      * The Julian time returned above is behind by one day, if "month" 
  752.      * or "year" is used to specify relative time and the GMT flag is true.
  753.      * This problem occurs only when the current time is closer to
  754.      * midnight, the difference being not more than its time difference
  755.      * with GMT. For example, in US/Pacific time zone, the problem occurs
  756.      * whenever the current time is between midnight to 8:00am or 7:00amDST.
  757.      * See Bug# 413397 for more details and sample script.
  758.      * To resolve this bug, we simply add the number of seconds corresponding
  759.      * to timezone difference with GMT to Julian time, if GMT flag is true.
  760.      */
  761.     if (TclDateTimezone == 0) {
  762.         Julian += TclpGetTimeZone((unsigned long) Start) * 60L;
  763.     }
  764.     /*
  765.      * The following iteration takes into account the case were we jump
  766.      * into a "short month".  Far example, "one month from Jan 31" will
  767.      * fail because there is no Feb 31.  The code below will reduce the
  768.      * day and try converting the date until we succed or the date equals
  769.      * 28 (which always works unless the date is bad in another way).
  770.      */
  771.     while ((result != 0) && (tm->tm_mday > 28)) {
  772. tm->tm_mday--;
  773. result = Convert(Month, (time_t) tm->tm_mday, Year,
  774. (time_t) tm->tm_hour, (time_t) tm->tm_min, (time_t) tm->tm_sec,
  775. MER24, DSTmaybe, &Julian);
  776.     }
  777.     if (result != 0) {
  778. return -1;
  779.     }
  780.     *TimePtr = DSTcorrect(Start, Julian);
  781.     return 0;
  782. }
  783. /*
  784.  *-----------------------------------------------------------------------------
  785.  *
  786.  * RelativeDay --
  787.  *
  788.  *      Given a starting time and a number of days before or after, compute the
  789.  *      DST corrected difference between those dates.
  790.  *
  791.  * Results:
  792.  *     1 or -1 indicating success or failure.
  793.  *
  794.  * Side effects:
  795.  *      Fills TimePtr with the computed value.
  796.  *
  797.  *-----------------------------------------------------------------------------
  798.  */
  799. static int
  800. RelativeDay(Start, RelDay, TimePtr)
  801.     time_t Start;
  802.     time_t RelDay;
  803.     time_t *TimePtr;
  804. {
  805.     time_t new;
  806.     new = Start + (RelDay * 60 * 60 * 24);
  807.     *TimePtr = DSTcorrect(Start, new);
  808.     return 1;
  809. }
  810. static int
  811. LookupWord(buff)
  812.     char                *buff;
  813. {
  814.     register char *p;
  815.     register char *q;
  816.     register TABLE *tp;
  817.     int i;
  818.     int abbrev;
  819.     /*
  820.      * Make it lowercase.
  821.      */
  822.     Tcl_UtfToLower(buff);
  823.     if (strcmp(buff, "am") == 0 || strcmp(buff, "a.m.") == 0) {
  824.         yylval.Meridian = MERam;
  825.         return tMERIDIAN;
  826.     }
  827.     if (strcmp(buff, "pm") == 0 || strcmp(buff, "p.m.") == 0) {
  828.         yylval.Meridian = MERpm;
  829.         return tMERIDIAN;
  830.     }
  831.     /*
  832.      * See if we have an abbreviation for a month.
  833.      */
  834.     if (strlen(buff) == 3) {
  835.         abbrev = 1;
  836.     } else if (strlen(buff) == 4 && buff[3] == '.') {
  837.         abbrev = 1;
  838.         buff[3] = '';
  839.     } else {
  840.         abbrev = 0;
  841.     }
  842.     for (tp = MonthDayTable; tp->name; tp++) {
  843.         if (abbrev) {
  844.             if (strncmp(buff, tp->name, 3) == 0) {
  845.                 yylval.Number = tp->value;
  846.                 return tp->type;
  847.             }
  848.         } else if (strcmp(buff, tp->name) == 0) {
  849.             yylval.Number = tp->value;
  850.             return tp->type;
  851.         }
  852.     }
  853.     for (tp = TimezoneTable; tp->name; tp++) {
  854.         if (strcmp(buff, tp->name) == 0) {
  855.             yylval.Number = tp->value;
  856.             return tp->type;
  857.         }
  858.     }
  859.     for (tp = UnitsTable; tp->name; tp++) {
  860.         if (strcmp(buff, tp->name) == 0) {
  861.             yylval.Number = tp->value;
  862.             return tp->type;
  863.         }
  864.     }
  865.     /*
  866.      * Strip off any plural and try the units table again.
  867.      */
  868.     i = strlen(buff) - 1;
  869.     if (buff[i] == 's') {
  870.         buff[i] = '';
  871.         for (tp = UnitsTable; tp->name; tp++) {
  872.             if (strcmp(buff, tp->name) == 0) {
  873.                 yylval.Number = tp->value;
  874.                 return tp->type;
  875.             }
  876. }
  877.     }
  878.     for (tp = OtherTable; tp->name; tp++) {
  879.         if (strcmp(buff, tp->name) == 0) {
  880.             yylval.Number = tp->value;
  881.             return tp->type;
  882.         }
  883.     }
  884.     /*
  885.      * Military timezones.
  886.      */
  887.     if (buff[1] == '' && !(*buff & 0x80)
  888.     && isalpha(UCHAR(*buff))) { /* INTL: ISO only */
  889.         for (tp = MilitaryTable; tp->name; tp++) {
  890.             if (strcmp(buff, tp->name) == 0) {
  891.                 yylval.Number = tp->value;
  892.                 return tp->type;
  893.             }
  894. }
  895.     }
  896.     /*
  897.      * Drop out any periods and try the timezone table again.
  898.      */
  899.     for (i = 0, p = q = buff; *q; q++)
  900.         if (*q != '.') {
  901.             *p++ = *q;
  902.         } else {
  903.             i++;
  904. }
  905.     *p = '';
  906.     if (i) {
  907.         for (tp = TimezoneTable; tp->name; tp++) {
  908.             if (strcmp(buff, tp->name) == 0) {
  909.                 yylval.Number = tp->value;
  910.                 return tp->type;
  911.             }
  912. }
  913.     }
  914.     
  915.     return tID;
  916. }
  917. static int
  918. yylex()
  919. {
  920.     register char       c;
  921.     register char       *p;
  922.     char                buff[20];
  923.     int                 Count;
  924.     for ( ; ; ) {
  925.         while (isspace(UCHAR(*yyInput))) {
  926.             yyInput++;
  927. }
  928.         if (isdigit(UCHAR(c = *yyInput))) { /* INTL: digit */
  929.     /* convert the string into a number; count the number of digits */
  930.     Count = 0;
  931.             for (yylval.Number = 0;
  932.     isdigit(UCHAR(c = *yyInput++)); ) { /* INTL: digit */
  933.                 yylval.Number = 10 * yylval.Number + c - '0';
  934. Count++;
  935.     }
  936.             yyInput--;
  937.     /* A number with 6 or more digits is considered an ISO 8601 base */
  938.     if (Count >= 6) {
  939. return tISOBASE;
  940.     } else {
  941. return tUNUMBER;
  942.     }
  943.         }
  944.         if (!(c & 0x80) && isalpha(UCHAR(c))) { /* INTL: ISO only. */
  945.             for (p = buff; isalpha(UCHAR(c = *yyInput++)) /* INTL: ISO only. */
  946.      || c == '.'; ) {
  947.                 if (p < &buff[sizeof buff - 1]) {
  948.                     *p++ = c;
  949. }
  950.     }
  951.             *p = '';
  952.             yyInput--;
  953.             return LookupWord(buff);
  954.         }
  955.         if (c != '(') {
  956.             return *yyInput++;
  957. }
  958.         Count = 0;
  959.         do {
  960.             c = *yyInput++;
  961.             if (c == '') {
  962.                 return c;
  963.     } else if (c == '(') {
  964.                 Count++;
  965.     } else if (c == ')') {
  966.                 Count--;
  967.     }
  968.         } while (Count > 0);
  969.     }
  970. }
  971. /*
  972.  * Specify zone is of -50000 to force GMT.  (This allows BST to work).
  973.  */
  974. int
  975. TclGetDate(p, now, zone, timePtr)
  976.     char *p;
  977.     Tcl_WideInt now;
  978.     long zone;
  979.     Tcl_WideInt *timePtr;
  980. {
  981.     struct tm *tm;
  982.     time_t Start;
  983.     time_t Time;
  984.     time_t tod;
  985.     int thisyear;
  986.     yyInput = p;
  987.     /* now has to be cast to a time_t for 64bit compliance */
  988.     Start = (time_t) now;
  989.     tm = TclpGetDate((TclpTime_t) &Start, (zone == -50000));
  990.     thisyear = tm->tm_year + TM_YEAR_BASE;
  991.     yyYear = thisyear;
  992.     yyMonth = tm->tm_mon + 1;
  993.     yyDay = tm->tm_mday;
  994.     yyTimezone = zone;
  995.     if (zone == -50000) {
  996.         yyDSTmode = DSToff;  /* assume GMT */
  997.         yyTimezone = 0;
  998.     } else {
  999.         yyDSTmode = DSTmaybe;
  1000.     }
  1001.     yyHour = 0;
  1002.     yyMinutes = 0;
  1003.     yySeconds = 0;
  1004.     yyMeridian = MER24;
  1005.     yyRelSeconds = 0;
  1006.     yyRelMonth = 0;
  1007.     yyRelDay = 0;
  1008.     yyRelPointer = NULL;
  1009.     yyHaveDate = 0;
  1010.     yyHaveDay = 0;
  1011.     yyHaveOrdinalMonth = 0;
  1012.     yyHaveRel = 0;
  1013.     yyHaveTime = 0;
  1014.     yyHaveZone = 0;
  1015.     if (yyparse() || yyHaveTime > 1 || yyHaveZone > 1 || yyHaveDate > 1 ||
  1016.     yyHaveDay > 1 || yyHaveOrdinalMonth > 1) {
  1017.         return -1;
  1018.     }
  1019.     
  1020.     if (yyHaveDate || yyHaveTime || yyHaveDay) {
  1021. if (TclDateYear < 0) {
  1022.     TclDateYear = -TclDateYear;
  1023. }
  1024. /*
  1025.  * The following line handles years that are specified using
  1026.  * only two digits.  The line of code below implements a policy
  1027.  * defined by the X/Open workgroup on the millinium rollover.
  1028.  * Note: some of those dates may not actually be valid on some
  1029.  * platforms.  The POSIX standard startes that the dates 70-99
  1030.  * shall refer to 1970-1999 and 00-38 shall refer to 2000-2038.
  1031.  * This later definition should work on all platforms.
  1032.  */
  1033. if (TclDateYear < 100) {
  1034.     if (TclDateYear >= 69) {
  1035. TclDateYear += 1900;
  1036.     } else {
  1037. TclDateYear += 2000;
  1038.     }
  1039. }
  1040. if (Convert(yyMonth, yyDay, yyYear, yyHour, yyMinutes, yySeconds,
  1041. yyMeridian, yyDSTmode, &Start) < 0) {
  1042.             return -1;
  1043. }
  1044.     } else {
  1045.         Start = (time_t) now;
  1046.         if (!yyHaveRel) {
  1047.             Start -= ((tm->tm_hour * 60L * 60L) +
  1048.     tm->tm_min * 60L) + tm->tm_sec;
  1049. }
  1050.     }
  1051.     Start += yyRelSeconds;
  1052.     if (RelativeMonth(Start, yyRelMonth, &Time) < 0) {
  1053.         return -1;
  1054.     }
  1055.     Start += Time;
  1056.     if (RelativeDay(Start, yyRelDay, &Time) < 0) {
  1057. return -1;
  1058.     }
  1059.     Start += Time;
  1060.     
  1061.     if (yyHaveDay && !yyHaveDate) {
  1062.         tod = NamedDay(Start, yyDayOrdinal, yyDayNumber);
  1063.         Start += tod;
  1064.     }
  1065.     if (yyHaveOrdinalMonth) {
  1066. tod = NamedMonth(Start, yyMonthOrdinal, yyMonth);
  1067. Start += tod;
  1068.     }
  1069.     
  1070.     *timePtr = Start;
  1071.     return 0;
  1072. }