• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  *******************************************************************************
3  * Copyright (C) 1997-2010, International Business Machines Corporation and
4  * others. All Rights Reserved.
5  *******************************************************************************
6  *
7  * File SIMPLETZ.H
8  *
9  * Modification History:
10  *
11  *   Date        Name        Description
12  *   12/05/96    clhuang     Creation.
13  *   04/21/97    aliu        Fixed miscellaneous bugs found by inspection and
14  *                           testing.
15  *   07/29/97    aliu        Ported source bodies back from Java version with
16  *                           numerous feature enhancements and bug fixes.
17  *   08/10/98    stephen     JDK 1.2 sync.
18  *   09/17/98    stephen     Fixed getOffset() for last hour of year and DST
19  *   12/02/99    aliu        Added TimeMode and constructor and setStart/EndRule
20  *                           methods that take TimeMode. Whitespace cleanup.
21  ********************************************************************************
22  */
23 
24 #include "unicode/utypeinfo.h"  // for 'typeid' to work
25 
26 #include "unicode/utypes.h"
27 
28 #if !UCONFIG_NO_FORMATTING
29 
30 #include "unicode/simpletz.h"
31 #include "unicode/gregocal.h"
32 #include "unicode/smpdtfmt.h"
33 
34 #include "gregoimp.h"
35 
36 U_NAMESPACE_BEGIN
37 
38 UOBJECT_DEFINE_RTTI_IMPLEMENTATION(SimpleTimeZone)
39 
40 // Use only for decodeStartRule() and decodeEndRule() where the year is not
41 // available. Set February to 29 days to accomodate rules with that date
42 // and day-of-week-on-or-before-that-date mode (DOW_LE_DOM_MODE).
43 // The compareToRule() method adjusts to February 28 in non-leap years.
44 //
45 // For actual getOffset() calculations, use Grego::monthLength() and
46 // Grego::previousMonthLength() which take leap years into account.
47 // We handle leap years assuming always
48 // Gregorian, since we know they didn't have daylight time when
49 // Gregorian calendar started.
50 const int8_t SimpleTimeZone::STATICMONTHLENGTH[] = {31,29,31,30,31,30,31,31,30,31,30,31};
51 
52 static const UChar DST_STR[] = {0x0028,0x0044,0x0053,0x0054,0x0029,0}; // "(DST)"
53 static const UChar STD_STR[] = {0x0028,0x0053,0x0054,0x0044,0x0029,0}; // "(STD)"
54 
55 
56 // *****************************************************************************
57 // class SimpleTimeZone
58 // *****************************************************************************
59 
60 
SimpleTimeZone(int32_t rawOffsetGMT,const UnicodeString & ID)61 SimpleTimeZone::SimpleTimeZone(int32_t rawOffsetGMT, const UnicodeString& ID)
62 :   BasicTimeZone(ID),
63     startMonth(0),
64     startDay(0),
65     startDayOfWeek(0),
66     startTime(0),
67     startTimeMode(WALL_TIME),
68     endTimeMode(WALL_TIME),
69     endMonth(0),
70     endDay(0),
71     endDayOfWeek(0),
72     endTime(0),
73     startYear(0),
74     rawOffset(rawOffsetGMT),
75     useDaylight(FALSE),
76     startMode(DOM_MODE),
77     endMode(DOM_MODE),
78     dstSavings(U_MILLIS_PER_HOUR)
79 {
80     clearTransitionRules();
81 }
82 
83 // -------------------------------------
84 
SimpleTimeZone(int32_t rawOffsetGMT,const UnicodeString & ID,int8_t savingsStartMonth,int8_t savingsStartDay,int8_t savingsStartDayOfWeek,int32_t savingsStartTime,int8_t savingsEndMonth,int8_t savingsEndDay,int8_t savingsEndDayOfWeek,int32_t savingsEndTime,UErrorCode & status)85 SimpleTimeZone::SimpleTimeZone(int32_t rawOffsetGMT, const UnicodeString& ID,
86     int8_t savingsStartMonth, int8_t savingsStartDay,
87     int8_t savingsStartDayOfWeek, int32_t savingsStartTime,
88     int8_t savingsEndMonth, int8_t savingsEndDay,
89     int8_t savingsEndDayOfWeek, int32_t savingsEndTime,
90     UErrorCode& status)
91 :   BasicTimeZone(ID)
92 {
93     clearTransitionRules();
94     construct(rawOffsetGMT,
95               savingsStartMonth, savingsStartDay, savingsStartDayOfWeek,
96               savingsStartTime, WALL_TIME,
97               savingsEndMonth, savingsEndDay, savingsEndDayOfWeek,
98               savingsEndTime, WALL_TIME,
99               U_MILLIS_PER_HOUR, status);
100 }
101 
102 // -------------------------------------
103 
SimpleTimeZone(int32_t rawOffsetGMT,const UnicodeString & ID,int8_t savingsStartMonth,int8_t savingsStartDay,int8_t savingsStartDayOfWeek,int32_t savingsStartTime,int8_t savingsEndMonth,int8_t savingsEndDay,int8_t savingsEndDayOfWeek,int32_t savingsEndTime,int32_t savingsDST,UErrorCode & status)104 SimpleTimeZone::SimpleTimeZone(int32_t rawOffsetGMT, const UnicodeString& ID,
105     int8_t savingsStartMonth, int8_t savingsStartDay,
106     int8_t savingsStartDayOfWeek, int32_t savingsStartTime,
107     int8_t savingsEndMonth, int8_t savingsEndDay,
108     int8_t savingsEndDayOfWeek, int32_t savingsEndTime,
109     int32_t savingsDST, UErrorCode& status)
110 :   BasicTimeZone(ID)
111 {
112     clearTransitionRules();
113     construct(rawOffsetGMT,
114               savingsStartMonth, savingsStartDay, savingsStartDayOfWeek,
115               savingsStartTime, WALL_TIME,
116               savingsEndMonth, savingsEndDay, savingsEndDayOfWeek,
117               savingsEndTime, WALL_TIME,
118               savingsDST, status);
119 }
120 
121 // -------------------------------------
122 
SimpleTimeZone(int32_t rawOffsetGMT,const UnicodeString & ID,int8_t savingsStartMonth,int8_t savingsStartDay,int8_t savingsStartDayOfWeek,int32_t savingsStartTime,TimeMode savingsStartTimeMode,int8_t savingsEndMonth,int8_t savingsEndDay,int8_t savingsEndDayOfWeek,int32_t savingsEndTime,TimeMode savingsEndTimeMode,int32_t savingsDST,UErrorCode & status)123 SimpleTimeZone::SimpleTimeZone(int32_t rawOffsetGMT, const UnicodeString& ID,
124     int8_t savingsStartMonth, int8_t savingsStartDay,
125     int8_t savingsStartDayOfWeek, int32_t savingsStartTime,
126     TimeMode savingsStartTimeMode,
127     int8_t savingsEndMonth, int8_t savingsEndDay,
128     int8_t savingsEndDayOfWeek, int32_t savingsEndTime,
129     TimeMode savingsEndTimeMode,
130     int32_t savingsDST, UErrorCode& status)
131 :   BasicTimeZone(ID)
132 {
133     clearTransitionRules();
134     construct(rawOffsetGMT,
135               savingsStartMonth, savingsStartDay, savingsStartDayOfWeek,
136               savingsStartTime, savingsStartTimeMode,
137               savingsEndMonth, savingsEndDay, savingsEndDayOfWeek,
138               savingsEndTime, savingsEndTimeMode,
139               savingsDST, status);
140 }
141 
142 /**
143  * Internal construction method.
144  */
construct(int32_t rawOffsetGMT,int8_t savingsStartMonth,int8_t savingsStartDay,int8_t savingsStartDayOfWeek,int32_t savingsStartTime,TimeMode savingsStartTimeMode,int8_t savingsEndMonth,int8_t savingsEndDay,int8_t savingsEndDayOfWeek,int32_t savingsEndTime,TimeMode savingsEndTimeMode,int32_t savingsDST,UErrorCode & status)145 void SimpleTimeZone::construct(int32_t rawOffsetGMT,
146                                int8_t savingsStartMonth,
147                                int8_t savingsStartDay,
148                                int8_t savingsStartDayOfWeek,
149                                int32_t savingsStartTime,
150                                TimeMode savingsStartTimeMode,
151                                int8_t savingsEndMonth,
152                                int8_t savingsEndDay,
153                                int8_t savingsEndDayOfWeek,
154                                int32_t savingsEndTime,
155                                TimeMode savingsEndTimeMode,
156                                int32_t savingsDST,
157                                UErrorCode& status)
158 {
159     this->rawOffset      = rawOffsetGMT;
160     this->startMonth     = savingsStartMonth;
161     this->startDay       = savingsStartDay;
162     this->startDayOfWeek = savingsStartDayOfWeek;
163     this->startTime      = savingsStartTime;
164     this->startTimeMode  = savingsStartTimeMode;
165     this->endMonth       = savingsEndMonth;
166     this->endDay         = savingsEndDay;
167     this->endDayOfWeek   = savingsEndDayOfWeek;
168     this->endTime        = savingsEndTime;
169     this->endTimeMode    = savingsEndTimeMode;
170     this->dstSavings     = savingsDST;
171     this->startYear      = 0;
172     this->startMode      = DOM_MODE;
173     this->endMode        = DOM_MODE;
174 
175     decodeRules(status);
176 
177     if (savingsDST <= 0) {
178         status = U_ILLEGAL_ARGUMENT_ERROR;
179     }
180 }
181 
182 // -------------------------------------
183 
~SimpleTimeZone()184 SimpleTimeZone::~SimpleTimeZone()
185 {
186     deleteTransitionRules();
187 }
188 
189 // -------------------------------------
190 
191 // Called by TimeZone::createDefault(), then clone() inside a Mutex - be careful.
SimpleTimeZone(const SimpleTimeZone & source)192 SimpleTimeZone::SimpleTimeZone(const SimpleTimeZone &source)
193 :   BasicTimeZone(source)
194 {
195     *this = source;
196 }
197 
198 // -------------------------------------
199 
200 // Called by TimeZone::createDefault(), then clone() inside a Mutex - be careful.
201 SimpleTimeZone &
operator =(const SimpleTimeZone & right)202 SimpleTimeZone::operator=(const SimpleTimeZone &right)
203 {
204     if (this != &right)
205     {
206         TimeZone::operator=(right);
207         rawOffset      = right.rawOffset;
208         startMonth     = right.startMonth;
209         startDay       = right.startDay;
210         startDayOfWeek = right.startDayOfWeek;
211         startTime      = right.startTime;
212         startTimeMode  = right.startTimeMode;
213         startMode      = right.startMode;
214         endMonth       = right.endMonth;
215         endDay         = right.endDay;
216         endDayOfWeek   = right.endDayOfWeek;
217         endTime        = right.endTime;
218         endTimeMode    = right.endTimeMode;
219         endMode        = right.endMode;
220         startYear      = right.startYear;
221         dstSavings     = right.dstSavings;
222         useDaylight    = right.useDaylight;
223         clearTransitionRules();
224     }
225     return *this;
226 }
227 
228 // -------------------------------------
229 
230 UBool
operator ==(const TimeZone & that) const231 SimpleTimeZone::operator==(const TimeZone& that) const
232 {
233     return ((this == &that) ||
234             (typeid(*this) == typeid(that) &&
235             TimeZone::operator==(that) &&
236             hasSameRules(that)));
237 }
238 
239 // -------------------------------------
240 
241 // Called by TimeZone::createDefault() inside a Mutex - be careful.
242 TimeZone*
clone() const243 SimpleTimeZone::clone() const
244 {
245     return new SimpleTimeZone(*this);
246 }
247 
248 // -------------------------------------
249 
250 /**
251  * Sets the daylight savings starting year, that is, the year this time zone began
252  * observing its specified daylight savings time rules.  The time zone is considered
253  * not to observe daylight savings time prior to that year; SimpleTimeZone doesn't
254  * support historical daylight-savings-time rules.
255  * @param year the daylight savings starting year.
256  */
257 void
setStartYear(int32_t year)258 SimpleTimeZone::setStartYear(int32_t year)
259 {
260     startYear = year;
261     transitionRulesInitialized = FALSE;
262 }
263 
264 // -------------------------------------
265 
266 /**
267  * Sets the daylight savings starting rule. For example, in the U.S., Daylight Savings
268  * Time starts at the first Sunday in April, at 2 AM in standard time.
269  * Therefore, you can set the start rule by calling:
270  * setStartRule(TimeFields.APRIL, 1, TimeFields.SUNDAY, 2*60*60*1000);
271  * The dayOfWeekInMonth and dayOfWeek parameters together specify how to calculate
272  * the exact starting date.  Their exact meaning depend on their respective signs,
273  * allowing various types of rules to be constructed, as follows:<ul>
274  *   <li>If both dayOfWeekInMonth and dayOfWeek are positive, they specify the
275  *       day of week in the month (e.g., (2, WEDNESDAY) is the second Wednesday
276  *       of the month).
277  *   <li>If dayOfWeek is positive and dayOfWeekInMonth is negative, they specify
278  *       the day of week in the month counting backward from the end of the month.
279  *       (e.g., (-1, MONDAY) is the last Monday in the month)
280  *   <li>If dayOfWeek is zero and dayOfWeekInMonth is positive, dayOfWeekInMonth
281  *       specifies the day of the month, regardless of what day of the week it is.
282  *       (e.g., (10, 0) is the tenth day of the month)
283  *   <li>If dayOfWeek is zero and dayOfWeekInMonth is negative, dayOfWeekInMonth
284  *       specifies the day of the month counting backward from the end of the
285  *       month, regardless of what day of the week it is (e.g., (-2, 0) is the
286  *       next-to-last day of the month).
287  *   <li>If dayOfWeek is negative and dayOfWeekInMonth is positive, they specify the
288  *       first specified day of the week on or after the specfied day of the month.
289  *       (e.g., (15, -SUNDAY) is the first Sunday after the 15th of the month
290  *       [or the 15th itself if the 15th is a Sunday].)
291  *   <li>If dayOfWeek and DayOfWeekInMonth are both negative, they specify the
292  *       last specified day of the week on or before the specified day of the month.
293  *       (e.g., (-20, -TUESDAY) is the last Tuesday before the 20th of the month
294  *       [or the 20th itself if the 20th is a Tuesday].)</ul>
295  * @param month the daylight savings starting month. Month is 0-based.
296  * eg, 0 for January.
297  * @param dayOfWeekInMonth the daylight savings starting
298  * day-of-week-in-month. Please see the member description for an example.
299  * @param dayOfWeek the daylight savings starting day-of-week. Please see
300  * the member description for an example.
301  * @param time the daylight savings starting time. Please see the member
302  * description for an example.
303  */
304 
305 void
setStartRule(int32_t month,int32_t dayOfWeekInMonth,int32_t dayOfWeek,int32_t time,TimeMode mode,UErrorCode & status)306 SimpleTimeZone::setStartRule(int32_t month, int32_t dayOfWeekInMonth, int32_t dayOfWeek,
307                              int32_t time, TimeMode mode, UErrorCode& status)
308 {
309     startMonth     = (int8_t)month;
310     startDay       = (int8_t)dayOfWeekInMonth;
311     startDayOfWeek = (int8_t)dayOfWeek;
312     startTime      = time;
313     startTimeMode  = mode;
314     decodeStartRule(status);
315     transitionRulesInitialized = FALSE;
316 }
317 
318 // -------------------------------------
319 
320 void
setStartRule(int32_t month,int32_t dayOfMonth,int32_t time,TimeMode mode,UErrorCode & status)321 SimpleTimeZone::setStartRule(int32_t month, int32_t dayOfMonth,
322                              int32_t time, TimeMode mode, UErrorCode& status)
323 {
324     setStartRule(month, dayOfMonth, 0, time, mode, status);
325 }
326 
327 // -------------------------------------
328 
329 void
setStartRule(int32_t month,int32_t dayOfMonth,int32_t dayOfWeek,int32_t time,TimeMode mode,UBool after,UErrorCode & status)330 SimpleTimeZone::setStartRule(int32_t month, int32_t dayOfMonth, int32_t dayOfWeek,
331                              int32_t time, TimeMode mode, UBool after, UErrorCode& status)
332 {
333     setStartRule(month, after ? dayOfMonth : -dayOfMonth,
334                  -dayOfWeek, time, mode, status);
335 }
336 
337 // -------------------------------------
338 
339 /**
340  * Sets the daylight savings ending rule. For example, in the U.S., Daylight
341  * Savings Time ends at the last (-1) Sunday in October, at 2 AM in standard time.
342  * Therefore, you can set the end rule by calling:
343  * setEndRule(TimeFields.OCTOBER, -1, TimeFields.SUNDAY, 2*60*60*1000);
344  * Various other types of rules can be specified by manipulating the dayOfWeek
345  * and dayOfWeekInMonth parameters.  For complete details, see the documentation
346  * for setStartRule().
347  * @param month the daylight savings ending month. Month is 0-based.
348  * eg, 0 for January.
349  * @param dayOfWeekInMonth the daylight savings ending
350  * day-of-week-in-month. See setStartRule() for a complete explanation.
351  * @param dayOfWeek the daylight savings ending day-of-week. See setStartRule()
352  * for a complete explanation.
353  * @param time the daylight savings ending time. Please see the member
354  * description for an example.
355  */
356 
357 void
setEndRule(int32_t month,int32_t dayOfWeekInMonth,int32_t dayOfWeek,int32_t time,TimeMode mode,UErrorCode & status)358 SimpleTimeZone::setEndRule(int32_t month, int32_t dayOfWeekInMonth, int32_t dayOfWeek,
359                            int32_t time, TimeMode mode, UErrorCode& status)
360 {
361     endMonth     = (int8_t)month;
362     endDay       = (int8_t)dayOfWeekInMonth;
363     endDayOfWeek = (int8_t)dayOfWeek;
364     endTime      = time;
365     endTimeMode  = mode;
366     decodeEndRule(status);
367     transitionRulesInitialized = FALSE;
368 }
369 
370 // -------------------------------------
371 
372 void
setEndRule(int32_t month,int32_t dayOfMonth,int32_t time,TimeMode mode,UErrorCode & status)373 SimpleTimeZone::setEndRule(int32_t month, int32_t dayOfMonth,
374                            int32_t time, TimeMode mode, UErrorCode& status)
375 {
376     setEndRule(month, dayOfMonth, 0, time, mode, status);
377 }
378 
379 // -------------------------------------
380 
381 void
setEndRule(int32_t month,int32_t dayOfMonth,int32_t dayOfWeek,int32_t time,TimeMode mode,UBool after,UErrorCode & status)382 SimpleTimeZone::setEndRule(int32_t month, int32_t dayOfMonth, int32_t dayOfWeek,
383                            int32_t time, TimeMode mode, UBool after, UErrorCode& status)
384 {
385     setEndRule(month, after ? dayOfMonth : -dayOfMonth,
386                -dayOfWeek, time, mode, status);
387 }
388 
389 // -------------------------------------
390 
391 int32_t
getOffset(uint8_t era,int32_t year,int32_t month,int32_t day,uint8_t dayOfWeek,int32_t millis,UErrorCode & status) const392 SimpleTimeZone::getOffset(uint8_t era, int32_t year, int32_t month, int32_t day,
393                           uint8_t dayOfWeek, int32_t millis, UErrorCode& status) const
394 {
395     // Check the month before calling Grego::monthLength(). This
396     // duplicates the test that occurs in the 7-argument getOffset(),
397     // however, this is unavoidable. We don't mind because this method, in
398     // fact, should not be called; internal code should always call the
399     // 7-argument getOffset(), and outside code should use Calendar.get(int
400     // field) with fields ZONE_OFFSET and DST_OFFSET. We can't get rid of
401     // this method because it's public API. - liu 8/10/98
402     if(month < UCAL_JANUARY || month > UCAL_DECEMBER) {
403         status = U_ILLEGAL_ARGUMENT_ERROR;
404         return 0;
405     }
406 
407     return getOffset(era, year, month, day, dayOfWeek, millis, Grego::monthLength(year, month), status);
408 }
409 
410 int32_t
getOffset(uint8_t era,int32_t year,int32_t month,int32_t day,uint8_t dayOfWeek,int32_t millis,int32_t,UErrorCode & status) const411 SimpleTimeZone::getOffset(uint8_t era, int32_t year, int32_t month, int32_t day,
412                           uint8_t dayOfWeek, int32_t millis,
413                           int32_t /*monthLength*/, UErrorCode& status) const
414 {
415     // Check the month before calling Grego::monthLength(). This
416     // duplicates a test that occurs in the 9-argument getOffset(),
417     // however, this is unavoidable. We don't mind because this method, in
418     // fact, should not be called; internal code should always call the
419     // 9-argument getOffset(), and outside code should use Calendar.get(int
420     // field) with fields ZONE_OFFSET and DST_OFFSET. We can't get rid of
421     // this method because it's public API. - liu 8/10/98
422     if (month < UCAL_JANUARY
423         || month > UCAL_DECEMBER) {
424         status = U_ILLEGAL_ARGUMENT_ERROR;
425         return -1;
426     }
427 
428     // We ignore monthLength because it can be derived from year and month.
429     // This is so that February in leap years is calculated correctly.
430     // We keep this argument in this function for backwards compatibility.
431     return getOffset(era, year, month, day, dayOfWeek, millis,
432                      Grego::monthLength(year, month),
433                      Grego::previousMonthLength(year, month),
434                      status);
435 }
436 
437 int32_t
getOffset(uint8_t era,int32_t year,int32_t month,int32_t day,uint8_t dayOfWeek,int32_t millis,int32_t monthLength,int32_t prevMonthLength,UErrorCode & status) const438 SimpleTimeZone::getOffset(uint8_t era, int32_t year, int32_t month, int32_t day,
439                           uint8_t dayOfWeek, int32_t millis,
440                           int32_t monthLength, int32_t prevMonthLength,
441                           UErrorCode& status) const
442 {
443     if(U_FAILURE(status)) return 0;
444 
445     if ((era != GregorianCalendar::AD && era != GregorianCalendar::BC)
446         || month < UCAL_JANUARY
447         || month > UCAL_DECEMBER
448         || day < 1
449         || day > monthLength
450         || dayOfWeek < UCAL_SUNDAY
451         || dayOfWeek > UCAL_SATURDAY
452         || millis < 0
453         || millis >= U_MILLIS_PER_DAY
454         || monthLength < 28
455         || monthLength > 31
456         || prevMonthLength < 28
457         || prevMonthLength > 31) {
458         status = U_ILLEGAL_ARGUMENT_ERROR;
459         return -1;
460     }
461 
462     int32_t result = rawOffset;
463 
464     // Bail out if we are before the onset of daylight savings time
465     if(!useDaylight || year < startYear || era != GregorianCalendar::AD)
466         return result;
467 
468     // Check for southern hemisphere.  We assume that the start and end
469     // month are different.
470     UBool southern = (startMonth > endMonth);
471 
472     // Compare the date to the starting and ending rules.+1 = date>rule, -1
473     // = date<rule, 0 = date==rule.
474     int32_t startCompare = compareToRule((int8_t)month, (int8_t)monthLength, (int8_t)prevMonthLength,
475                                          (int8_t)day, (int8_t)dayOfWeek, millis,
476                                          startTimeMode == UTC_TIME ? -rawOffset : 0,
477                                          startMode, (int8_t)startMonth, (int8_t)startDayOfWeek,
478                                          (int8_t)startDay, startTime);
479     int32_t endCompare = 0;
480 
481     /* We don't always have to compute endCompare.  For many instances,
482      * startCompare is enough to determine if we are in DST or not.  In the
483      * northern hemisphere, if we are before the start rule, we can't have
484      * DST.  In the southern hemisphere, if we are after the start rule, we
485      * must have DST.  This is reflected in the way the next if statement
486      * (not the one immediately following) short circuits. */
487     if(southern != (startCompare >= 0)) {
488         endCompare = compareToRule((int8_t)month, (int8_t)monthLength, (int8_t)prevMonthLength,
489                                    (int8_t)day, (int8_t)dayOfWeek, millis,
490                                    endTimeMode == WALL_TIME ? dstSavings :
491                                     (endTimeMode == UTC_TIME ? -rawOffset : 0),
492                                    endMode, (int8_t)endMonth, (int8_t)endDayOfWeek,
493                                    (int8_t)endDay, endTime);
494     }
495 
496     // Check for both the northern and southern hemisphere cases.  We
497     // assume that in the northern hemisphere, the start rule is before the
498     // end rule within the calendar year, and vice versa for the southern
499     // hemisphere.
500     if ((!southern && (startCompare >= 0 && endCompare < 0)) ||
501         (southern && (startCompare >= 0 || endCompare < 0)))
502         result += dstSavings;
503 
504     return result;
505 }
506 
507 void
getOffsetFromLocal(UDate date,int32_t nonExistingTimeOpt,int32_t duplicatedTimeOpt,int32_t & rawOffsetGMT,int32_t & savingsDST,UErrorCode & status)508 SimpleTimeZone::getOffsetFromLocal(UDate date, int32_t nonExistingTimeOpt, int32_t duplicatedTimeOpt,
509                                    int32_t& rawOffsetGMT, int32_t& savingsDST, UErrorCode& status) /*const*/ {
510     if (U_FAILURE(status)) {
511         return;
512     }
513 
514     rawOffsetGMT = getRawOffset();
515     int32_t year, month, dom, dow;
516     double day = uprv_floor(date / U_MILLIS_PER_DAY);
517     int32_t millis = (int32_t) (date - day * U_MILLIS_PER_DAY);
518 
519     Grego::dayToFields(day, year, month, dom, dow);
520 
521     savingsDST = getOffset(GregorianCalendar::AD, year, month, dom,
522                           (uint8_t) dow, millis,
523                           Grego::monthLength(year, month),
524                           status) - rawOffsetGMT;
525     if (U_FAILURE(status)) {
526         return;
527     }
528 
529     UBool recalc = FALSE;
530 
531     // Now we need some adjustment
532     if (savingsDST > 0) {
533         if ((nonExistingTimeOpt & kStdDstMask) == kStandard
534             || ((nonExistingTimeOpt & kStdDstMask) != kDaylight && (nonExistingTimeOpt & kFormerLatterMask) != kLatter)) {
535             date -= getDSTSavings();
536             recalc = TRUE;
537         }
538     } else {
539         if ((duplicatedTimeOpt & kStdDstMask) == kDaylight
540                 || ((duplicatedTimeOpt & kStdDstMask) != kStandard && (duplicatedTimeOpt & kFormerLatterMask) == kFormer)) {
541             date -= getDSTSavings();
542             recalc = TRUE;
543         }
544     }
545     if (recalc) {
546         day = uprv_floor(date / U_MILLIS_PER_DAY);
547         millis = (int32_t) (date - day * U_MILLIS_PER_DAY);
548         Grego::dayToFields(day, year, month, dom, dow);
549         savingsDST = getOffset(GregorianCalendar::AD, year, month, dom,
550                           (uint8_t) dow, millis,
551                           Grego::monthLength(year, month),
552                           status) - rawOffsetGMT;
553     }
554 }
555 
556 // -------------------------------------
557 
558 /**
559  * Compare a given date in the year to a rule. Return 1, 0, or -1, depending
560  * on whether the date is after, equal to, or before the rule date. The
561  * millis are compared directly against the ruleMillis, so any
562  * standard-daylight adjustments must be handled by the caller.
563  *
564  * @return  1 if the date is after the rule date, -1 if the date is before
565  *          the rule date, or 0 if the date is equal to the rule date.
566  */
567 int32_t
compareToRule(int8_t month,int8_t monthLen,int8_t prevMonthLen,int8_t dayOfMonth,int8_t dayOfWeek,int32_t millis,int32_t millisDelta,EMode ruleMode,int8_t ruleMonth,int8_t ruleDayOfWeek,int8_t ruleDay,int32_t ruleMillis)568 SimpleTimeZone::compareToRule(int8_t month, int8_t monthLen, int8_t prevMonthLen,
569                               int8_t dayOfMonth,
570                               int8_t dayOfWeek, int32_t millis, int32_t millisDelta,
571                               EMode ruleMode, int8_t ruleMonth, int8_t ruleDayOfWeek,
572                               int8_t ruleDay, int32_t ruleMillis)
573 {
574     // Make adjustments for startTimeMode and endTimeMode
575     millis += millisDelta;
576     while (millis >= U_MILLIS_PER_DAY) {
577         millis -= U_MILLIS_PER_DAY;
578         ++dayOfMonth;
579         dayOfWeek = (int8_t)(1 + (dayOfWeek % 7)); // dayOfWeek is one-based
580         if (dayOfMonth > monthLen) {
581             dayOfMonth = 1;
582             /* When incrementing the month, it is desirible to overflow
583              * from DECEMBER to DECEMBER+1, since we use the result to
584              * compare against a real month. Wraparound of the value
585              * leads to bug 4173604. */
586             ++month;
587         }
588     }
589     while (millis < 0) {
590         millis += U_MILLIS_PER_DAY;
591         --dayOfMonth;
592         dayOfWeek = (int8_t)(1 + ((dayOfWeek+5) % 7)); // dayOfWeek is one-based
593         if (dayOfMonth < 1) {
594             dayOfMonth = prevMonthLen;
595             --month;
596         }
597     }
598 
599     // first compare months.  If they're different, we don't have to worry about days
600     // and times
601     if (month < ruleMonth) return -1;
602     else if (month > ruleMonth) return 1;
603 
604     // calculate the actual day of month for the rule
605     int32_t ruleDayOfMonth = 0;
606 
607     // Adjust the ruleDay to the monthLen, for non-leap year February 29 rule days.
608     if (ruleDay > monthLen) {
609         ruleDay = monthLen;
610     }
611 
612     switch (ruleMode)
613     {
614     // if the mode is day-of-month, the day of month is given
615     case DOM_MODE:
616         ruleDayOfMonth = ruleDay;
617         break;
618 
619     // if the mode is day-of-week-in-month, calculate the day-of-month from it
620     case DOW_IN_MONTH_MODE:
621         // In this case ruleDay is the day-of-week-in-month (this code is using
622         // the dayOfWeek and dayOfMonth parameters to figure out the day-of-week
623         // of the first day of the month, so it's trusting that they're really
624         // consistent with each other)
625         if (ruleDay > 0)
626             ruleDayOfMonth = 1 + (ruleDay - 1) * 7 +
627                 (7 + ruleDayOfWeek - (dayOfWeek - dayOfMonth + 1)) % 7;
628 
629         // if ruleDay is negative (we assume it's not zero here), we have to do
630         // the same calculation figuring backward from the last day of the month.
631         else
632         {
633             // (again, this code is trusting that dayOfWeek and dayOfMonth are
634             // consistent with each other here, since we're using them to figure
635             // the day of week of the first of the month)
636             ruleDayOfMonth = monthLen + (ruleDay + 1) * 7 -
637                 (7 + (dayOfWeek + monthLen - dayOfMonth) - ruleDayOfWeek) % 7;
638         }
639         break;
640 
641     case DOW_GE_DOM_MODE:
642         ruleDayOfMonth = ruleDay +
643             (49 + ruleDayOfWeek - ruleDay - dayOfWeek + dayOfMonth) % 7;
644         break;
645 
646     case DOW_LE_DOM_MODE:
647         ruleDayOfMonth = ruleDay -
648             (49 - ruleDayOfWeek + ruleDay + dayOfWeek - dayOfMonth) % 7;
649         // Note at this point ruleDayOfMonth may be <1, although it will
650         // be >=1 for well-formed rules.
651         break;
652     }
653 
654     // now that we have a real day-in-month for the rule, we can compare days...
655     if (dayOfMonth < ruleDayOfMonth) return -1;
656     else if (dayOfMonth > ruleDayOfMonth) return 1;
657 
658     // ...and if they're equal, we compare times
659     if (millis < ruleMillis) return -1;
660     else if (millis > ruleMillis) return 1;
661     else return 0;
662 }
663 
664 // -------------------------------------
665 
666 int32_t
getRawOffset() const667 SimpleTimeZone::getRawOffset() const
668 {
669     return rawOffset;
670 }
671 
672 // -------------------------------------
673 
674 void
setRawOffset(int32_t offsetMillis)675 SimpleTimeZone::setRawOffset(int32_t offsetMillis)
676 {
677     rawOffset = offsetMillis;
678     transitionRulesInitialized = FALSE;
679 }
680 
681 // -------------------------------------
682 
683 void
setDSTSavings(int32_t millisSavedDuringDST,UErrorCode & status)684 SimpleTimeZone::setDSTSavings(int32_t millisSavedDuringDST, UErrorCode& status)
685 {
686     if (millisSavedDuringDST <= 0) {
687         status = U_ILLEGAL_ARGUMENT_ERROR;
688     }
689     else {
690         dstSavings = millisSavedDuringDST;
691     }
692     transitionRulesInitialized = FALSE;
693 }
694 
695 // -------------------------------------
696 
697 int32_t
getDSTSavings() const698 SimpleTimeZone::getDSTSavings() const
699 {
700     return dstSavings;
701 }
702 
703 // -------------------------------------
704 
705 UBool
useDaylightTime() const706 SimpleTimeZone::useDaylightTime() const
707 {
708     return useDaylight;
709 }
710 
711 // -------------------------------------
712 
713 /**
714  * Overrides TimeZone
715  * Queries if the given date is in Daylight Savings Time.
716  */
inDaylightTime(UDate date,UErrorCode & status) const717 UBool SimpleTimeZone::inDaylightTime(UDate date, UErrorCode& status) const
718 {
719     // This method is wasteful since it creates a new GregorianCalendar and
720     // deletes it each time it is called.  However, this is a deprecated method
721     // and provided only for Java compatibility as of 8/6/97 [LIU].
722     if (U_FAILURE(status)) return FALSE;
723     GregorianCalendar *gc = new GregorianCalendar(*this, status);
724     /* test for NULL */
725     if (gc == 0) {
726         status = U_MEMORY_ALLOCATION_ERROR;
727         return FALSE;
728     }
729     gc->setTime(date, status);
730     UBool result = gc->inDaylightTime(status);
731     delete gc;
732     return result;
733 }
734 
735 // -------------------------------------
736 
737 /**
738  * Return true if this zone has the same rules and offset as another zone.
739  * @param other the TimeZone object to be compared with
740  * @return true if the given zone has the same rules and offset as this one
741  */
742 UBool
hasSameRules(const TimeZone & other) const743 SimpleTimeZone::hasSameRules(const TimeZone& other) const
744 {
745     if (this == &other) return TRUE;
746     if (typeid(*this) != typeid(other)) return FALSE;
747     SimpleTimeZone *that = (SimpleTimeZone*)&other;
748     return rawOffset     == that->rawOffset &&
749         useDaylight     == that->useDaylight &&
750         (!useDaylight
751          // Only check rules if using DST
752          || (dstSavings     == that->dstSavings &&
753              startMode      == that->startMode &&
754              startMonth     == that->startMonth &&
755              startDay       == that->startDay &&
756              startDayOfWeek == that->startDayOfWeek &&
757              startTime      == that->startTime &&
758              startTimeMode  == that->startTimeMode &&
759              endMode        == that->endMode &&
760              endMonth       == that->endMonth &&
761              endDay         == that->endDay &&
762              endDayOfWeek   == that->endDayOfWeek &&
763              endTime        == that->endTime &&
764              endTimeMode    == that->endTimeMode &&
765              startYear      == that->startYear));
766 }
767 
768 // -------------------------------------
769 
770 //----------------------------------------------------------------------
771 // Rule representation
772 //
773 // We represent the following flavors of rules:
774 //       5        the fifth of the month
775 //       lastSun  the last Sunday in the month
776 //       lastMon  the last Monday in the month
777 //       Sun>=8   first Sunday on or after the eighth
778 //       Sun<=25  last Sunday on or before the 25th
779 // This is further complicated by the fact that we need to remain
780 // backward compatible with the 1.1 FCS.  Finally, we need to minimize
781 // API changes.  In order to satisfy these requirements, we support
782 // three representation systems, and we translate between them.
783 //
784 // INTERNAL REPRESENTATION
785 // This is the format SimpleTimeZone objects take after construction or
786 // streaming in is complete.  Rules are represented directly, using an
787 // unencoded format.  We will discuss the start rule only below; the end
788 // rule is analogous.
789 //   startMode      Takes on enumerated values DAY_OF_MONTH,
790 //                  DOW_IN_MONTH, DOW_AFTER_DOM, or DOW_BEFORE_DOM.
791 //   startDay       The day of the month, or for DOW_IN_MONTH mode, a
792 //                  value indicating which DOW, such as +1 for first,
793 //                  +2 for second, -1 for last, etc.
794 //   startDayOfWeek The day of the week.  Ignored for DAY_OF_MONTH.
795 //
796 // ENCODED REPRESENTATION
797 // This is the format accepted by the constructor and by setStartRule()
798 // and setEndRule().  It uses various combinations of positive, negative,
799 // and zero values to encode the different rules.  This representation
800 // allows us to specify all the different rule flavors without altering
801 // the API.
802 //   MODE              startMonth    startDay    startDayOfWeek
803 //   DOW_IN_MONTH_MODE >=0           !=0         >0
804 //   DOM_MODE          >=0           >0          ==0
805 //   DOW_GE_DOM_MODE   >=0           >0          <0
806 //   DOW_LE_DOM_MODE   >=0           <0          <0
807 //   (no DST)          don't care    ==0         don't care
808 //
809 // STREAMED REPRESENTATION
810 // We must retain binary compatibility with the 1.1 FCS.  The 1.1 code only
811 // handles DOW_IN_MONTH_MODE and non-DST mode, the latter indicated by the
812 // flag useDaylight.  When we stream an object out, we translate into an
813 // approximate DOW_IN_MONTH_MODE representation so the object can be parsed
814 // and used by 1.1 code.  Following that, we write out the full
815 // representation separately so that contemporary code can recognize and
816 // parse it.  The full representation is written in a "packed" format,
817 // consisting of a version number, a length, and an array of bytes.  Future
818 // versions of this class may specify different versions.  If they wish to
819 // include additional data, they should do so by storing them after the
820 // packed representation below.
821 //----------------------------------------------------------------------
822 
823 /**
824  * Given a set of encoded rules in startDay and startDayOfMonth, decode
825  * them and set the startMode appropriately.  Do the same for endDay and
826  * endDayOfMonth.  Upon entry, the day of week variables may be zero or
827  * negative, in order to indicate special modes.  The day of month
828  * variables may also be negative.  Upon exit, the mode variables will be
829  * set, and the day of week and day of month variables will be positive.
830  * This method also recognizes a startDay or endDay of zero as indicating
831  * no DST.
832  */
833 void
decodeRules(UErrorCode & status)834 SimpleTimeZone::decodeRules(UErrorCode& status)
835 {
836     decodeStartRule(status);
837     decodeEndRule(status);
838 }
839 
840 /**
841  * Decode the start rule and validate the parameters.  The parameters are
842  * expected to be in encoded form, which represents the various rule modes
843  * by negating or zeroing certain values.  Representation formats are:
844  * <p>
845  * <pre>
846  *            DOW_IN_MONTH  DOM    DOW>=DOM  DOW<=DOM  no DST
847  *            ------------  -----  --------  --------  ----------
848  * month       0..11        same    same      same     don't care
849  * day        -5..5         1..31   1..31    -1..-31   0
850  * dayOfWeek   1..7         0      -1..-7    -1..-7    don't care
851  * time        0..ONEDAY    same    same      same     don't care
852  * </pre>
853  * The range for month does not include UNDECIMBER since this class is
854  * really specific to GregorianCalendar, which does not use that month.
855  * The range for time includes ONEDAY (vs. ending at ONEDAY-1) because the
856  * end rule is an exclusive limit point.  That is, the range of times that
857  * are in DST include those >= the start and < the end.  For this reason,
858  * it should be possible to specify an end of ONEDAY in order to include the
859  * entire day.  Although this is equivalent to time 0 of the following day,
860  * it's not always possible to specify that, for example, on December 31.
861  * While arguably the start range should still be 0..ONEDAY-1, we keep
862  * the start and end ranges the same for consistency.
863  */
864 void
decodeStartRule(UErrorCode & status)865 SimpleTimeZone::decodeStartRule(UErrorCode& status)
866 {
867     if(U_FAILURE(status)) return;
868 
869     useDaylight = (UBool)((startDay != 0) && (endDay != 0) ? TRUE : FALSE);
870     if (useDaylight && dstSavings == 0) {
871         dstSavings = U_MILLIS_PER_HOUR;
872     }
873     if (startDay != 0) {
874         if (startMonth < UCAL_JANUARY || startMonth > UCAL_DECEMBER) {
875             status = U_ILLEGAL_ARGUMENT_ERROR;
876             return;
877         }
878         if (startTime < 0 || startTime > U_MILLIS_PER_DAY ||
879             startTimeMode < WALL_TIME || startTimeMode > UTC_TIME) {
880             status = U_ILLEGAL_ARGUMENT_ERROR;
881             return;
882         }
883         if (startDayOfWeek == 0) {
884             startMode = DOM_MODE;
885         } else {
886             if (startDayOfWeek > 0) {
887                 startMode = DOW_IN_MONTH_MODE;
888             } else {
889                 startDayOfWeek = (int8_t)-startDayOfWeek;
890                 if (startDay > 0) {
891                     startMode = DOW_GE_DOM_MODE;
892                 } else {
893                     startDay = (int8_t)-startDay;
894                     startMode = DOW_LE_DOM_MODE;
895                 }
896             }
897             if (startDayOfWeek > UCAL_SATURDAY) {
898                 status = U_ILLEGAL_ARGUMENT_ERROR;
899                 return;
900             }
901         }
902         if (startMode == DOW_IN_MONTH_MODE) {
903             if (startDay < -5 || startDay > 5) {
904                 status = U_ILLEGAL_ARGUMENT_ERROR;
905                 return;
906             }
907         } else if (startDay<1 || startDay > STATICMONTHLENGTH[startMonth]) {
908             status = U_ILLEGAL_ARGUMENT_ERROR;
909             return;
910         }
911     }
912 }
913 
914 /**
915  * Decode the end rule and validate the parameters.  This method is exactly
916  * analogous to decodeStartRule().
917  * @see decodeStartRule
918  */
919 void
decodeEndRule(UErrorCode & status)920 SimpleTimeZone::decodeEndRule(UErrorCode& status)
921 {
922     if(U_FAILURE(status)) return;
923 
924     useDaylight = (UBool)((startDay != 0) && (endDay != 0) ? TRUE : FALSE);
925     if (useDaylight && dstSavings == 0) {
926         dstSavings = U_MILLIS_PER_HOUR;
927     }
928     if (endDay != 0) {
929         if (endMonth < UCAL_JANUARY || endMonth > UCAL_DECEMBER) {
930             status = U_ILLEGAL_ARGUMENT_ERROR;
931             return;
932         }
933         if (endTime < 0 || endTime > U_MILLIS_PER_DAY ||
934             endTimeMode < WALL_TIME || endTimeMode > UTC_TIME) {
935             status = U_ILLEGAL_ARGUMENT_ERROR;
936             return;
937         }
938         if (endDayOfWeek == 0) {
939             endMode = DOM_MODE;
940         } else {
941             if (endDayOfWeek > 0) {
942                 endMode = DOW_IN_MONTH_MODE;
943             } else {
944                 endDayOfWeek = (int8_t)-endDayOfWeek;
945                 if (endDay > 0) {
946                     endMode = DOW_GE_DOM_MODE;
947                 } else {
948                     endDay = (int8_t)-endDay;
949                     endMode = DOW_LE_DOM_MODE;
950                 }
951             }
952             if (endDayOfWeek > UCAL_SATURDAY) {
953                 status = U_ILLEGAL_ARGUMENT_ERROR;
954                 return;
955             }
956         }
957         if (endMode == DOW_IN_MONTH_MODE) {
958             if (endDay < -5 || endDay > 5) {
959                 status = U_ILLEGAL_ARGUMENT_ERROR;
960                 return;
961             }
962         } else if (endDay<1 || endDay > STATICMONTHLENGTH[endMonth]) {
963             status = U_ILLEGAL_ARGUMENT_ERROR;
964             return;
965         }
966     }
967 }
968 
969 UBool
getNextTransition(UDate base,UBool inclusive,TimeZoneTransition & result)970 SimpleTimeZone::getNextTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ {
971     if (!useDaylight) {
972         return FALSE;
973     }
974 
975     UErrorCode status = U_ZERO_ERROR;
976     initTransitionRules(status);
977     if (U_FAILURE(status)) {
978         return FALSE;
979     }
980 
981     UDate firstTransitionTime = firstTransition->getTime();
982     if (base < firstTransitionTime || (inclusive && base == firstTransitionTime)) {
983         result = *firstTransition;
984     }
985     UDate stdDate, dstDate;
986     UBool stdAvail = stdRule->getNextStart(base, dstRule->getRawOffset(), dstRule->getDSTSavings(), inclusive, stdDate);
987     UBool dstAvail = dstRule->getNextStart(base, stdRule->getRawOffset(), stdRule->getDSTSavings(), inclusive, dstDate);
988     if (stdAvail && (!dstAvail || stdDate < dstDate)) {
989         result.setTime(stdDate);
990         result.setFrom((const TimeZoneRule&)*dstRule);
991         result.setTo((const TimeZoneRule&)*stdRule);
992         return TRUE;
993     }
994     if (dstAvail && (!stdAvail || dstDate < stdDate)) {
995         result.setTime(dstDate);
996         result.setFrom((const TimeZoneRule&)*stdRule);
997         result.setTo((const TimeZoneRule&)*dstRule);
998         return TRUE;
999     }
1000     return FALSE;
1001 }
1002 
1003 UBool
getPreviousTransition(UDate base,UBool inclusive,TimeZoneTransition & result)1004 SimpleTimeZone::getPreviousTransition(UDate base, UBool inclusive, TimeZoneTransition& result) /*const*/ {
1005     if (!useDaylight) {
1006         return FALSE;
1007     }
1008 
1009     UErrorCode status = U_ZERO_ERROR;
1010     initTransitionRules(status);
1011     if (U_FAILURE(status)) {
1012         return FALSE;
1013     }
1014 
1015     UDate firstTransitionTime = firstTransition->getTime();
1016     if (base < firstTransitionTime || (!inclusive && base == firstTransitionTime)) {
1017         return FALSE;
1018     }
1019     UDate stdDate, dstDate;
1020     UBool stdAvail = stdRule->getPreviousStart(base, dstRule->getRawOffset(), dstRule->getDSTSavings(), inclusive, stdDate);
1021     UBool dstAvail = dstRule->getPreviousStart(base, stdRule->getRawOffset(), stdRule->getDSTSavings(), inclusive, dstDate);
1022     if (stdAvail && (!dstAvail || stdDate > dstDate)) {
1023         result.setTime(stdDate);
1024         result.setFrom((const TimeZoneRule&)*dstRule);
1025         result.setTo((const TimeZoneRule&)*stdRule);
1026         return TRUE;
1027     }
1028     if (dstAvail && (!stdAvail || dstDate > stdDate)) {
1029         result.setTime(dstDate);
1030         result.setFrom((const TimeZoneRule&)*stdRule);
1031         result.setTo((const TimeZoneRule&)*dstRule);
1032         return TRUE;
1033     }
1034     return FALSE;
1035 }
1036 
1037 void
clearTransitionRules(void)1038 SimpleTimeZone::clearTransitionRules(void) {
1039     initialRule = NULL;
1040     firstTransition = NULL;
1041     stdRule = NULL;
1042     dstRule = NULL;
1043     transitionRulesInitialized = FALSE;
1044 }
1045 
1046 void
deleteTransitionRules(void)1047 SimpleTimeZone::deleteTransitionRules(void) {
1048     if (initialRule != NULL) {
1049         delete initialRule;
1050     }
1051     if (firstTransition != NULL) {
1052         delete firstTransition;
1053     }
1054     if (stdRule != NULL) {
1055         delete stdRule;
1056     }
1057     if (dstRule != NULL) {
1058         delete dstRule;
1059     }
1060     clearTransitionRules();
1061  }
1062 
1063 void
initTransitionRules(UErrorCode & status)1064 SimpleTimeZone::initTransitionRules(UErrorCode& status) {
1065     if (U_FAILURE(status)) {
1066         return;
1067     }
1068     if (transitionRulesInitialized) {
1069         return;
1070     }
1071     deleteTransitionRules();
1072     UnicodeString tzid;
1073     getID(tzid);
1074 
1075     if (useDaylight) {
1076         DateTimeRule* dtRule;
1077         DateTimeRule::TimeRuleType timeRuleType;
1078         UDate firstStdStart, firstDstStart;
1079 
1080         // Create a TimeZoneRule for daylight saving time
1081         timeRuleType = (startTimeMode == STANDARD_TIME) ? DateTimeRule::STANDARD_TIME :
1082             ((startTimeMode == UTC_TIME) ? DateTimeRule::UTC_TIME : DateTimeRule::WALL_TIME);
1083         switch (startMode) {
1084         case DOM_MODE:
1085             dtRule = new DateTimeRule(startMonth, startDay, startTime, timeRuleType);
1086             break;
1087         case DOW_IN_MONTH_MODE:
1088             dtRule = new DateTimeRule(startMonth, startDay, startDayOfWeek, startTime, timeRuleType);
1089             break;
1090         case DOW_GE_DOM_MODE:
1091             dtRule = new DateTimeRule(startMonth, startDay, startDayOfWeek, true, startTime, timeRuleType);
1092             break;
1093         case DOW_LE_DOM_MODE:
1094             dtRule = new DateTimeRule(startMonth, startDay, startDayOfWeek, false, startTime, timeRuleType);
1095             break;
1096         default:
1097             status = U_INVALID_STATE_ERROR;
1098             return;
1099         }
1100         // Check for Null pointer
1101         if (dtRule == NULL) {
1102         	status = U_MEMORY_ALLOCATION_ERROR;
1103         	return;
1104         }
1105         // For now, use ID + "(DST)" as the name
1106         dstRule = new AnnualTimeZoneRule(tzid+DST_STR, getRawOffset(), getDSTSavings(),
1107             dtRule, startYear, AnnualTimeZoneRule::MAX_YEAR);
1108 
1109         // Check for Null pointer
1110         if (dstRule == NULL) {
1111         	status = U_MEMORY_ALLOCATION_ERROR;
1112         	deleteTransitionRules();
1113         	return;
1114         }
1115 
1116         // Calculate the first DST start time
1117         dstRule->getFirstStart(getRawOffset(), 0, firstDstStart);
1118 
1119         // Create a TimeZoneRule for standard time
1120         timeRuleType = (endTimeMode == STANDARD_TIME) ? DateTimeRule::STANDARD_TIME :
1121             ((endTimeMode == UTC_TIME) ? DateTimeRule::UTC_TIME : DateTimeRule::WALL_TIME);
1122         switch (endMode) {
1123         case DOM_MODE:
1124             dtRule = new DateTimeRule(endMonth, endDay, endTime, timeRuleType);
1125             break;
1126         case DOW_IN_MONTH_MODE:
1127             dtRule = new DateTimeRule(endMonth, endDay, endDayOfWeek, endTime, timeRuleType);
1128             break;
1129         case DOW_GE_DOM_MODE:
1130             dtRule = new DateTimeRule(endMonth, endDay, endDayOfWeek, true, endTime, timeRuleType);
1131             break;
1132         case DOW_LE_DOM_MODE:
1133             dtRule = new DateTimeRule(endMonth, endDay, endDayOfWeek, false, endTime, timeRuleType);
1134             break;
1135         }
1136 
1137         // Check for Null pointer
1138         if (dtRule == NULL) {
1139         	status = U_MEMORY_ALLOCATION_ERROR;
1140         	deleteTransitionRules();
1141         	return;
1142         }
1143         // For now, use ID + "(STD)" as the name
1144         stdRule = new AnnualTimeZoneRule(tzid+STD_STR, getRawOffset(), 0,
1145             dtRule, startYear, AnnualTimeZoneRule::MAX_YEAR);
1146 
1147         //Check for Null pointer
1148         if (stdRule == NULL) {
1149         	status = U_MEMORY_ALLOCATION_ERROR;
1150         	deleteTransitionRules();
1151         	return;
1152         }
1153 
1154         // Calculate the first STD start time
1155         stdRule->getFirstStart(getRawOffset(), dstRule->getDSTSavings(), firstStdStart);
1156 
1157         // Create a TimeZoneRule for initial time
1158         if (firstStdStart < firstDstStart) {
1159             initialRule = new InitialTimeZoneRule(tzid+DST_STR, getRawOffset(), dstRule->getDSTSavings());
1160             firstTransition = new TimeZoneTransition(firstStdStart, *initialRule, *stdRule);
1161         } else {
1162             initialRule = new InitialTimeZoneRule(tzid+STD_STR, getRawOffset(), 0);
1163             firstTransition = new TimeZoneTransition(firstDstStart, *initialRule, *dstRule);
1164         }
1165         // Check for null pointers.
1166         if (initialRule == NULL || firstTransition == NULL) {
1167         	status = U_MEMORY_ALLOCATION_ERROR;
1168         	deleteTransitionRules();
1169         	return;
1170         }
1171 
1172     } else {
1173         // Create a TimeZoneRule for initial time
1174         initialRule = new InitialTimeZoneRule(tzid, getRawOffset(), 0);
1175         // Check for null pointer.
1176         if (initialRule == NULL) {
1177         	status = U_MEMORY_ALLOCATION_ERROR;
1178         	deleteTransitionRules();
1179         	return;
1180         }
1181     }
1182 
1183     transitionRulesInitialized = true;
1184 }
1185 
1186 int32_t
countTransitionRules(UErrorCode &)1187 SimpleTimeZone::countTransitionRules(UErrorCode& /*status*/) /*const*/ {
1188     return (useDaylight) ? 2 : 0;
1189 }
1190 
1191 void
getTimeZoneRules(const InitialTimeZoneRule * & initial,const TimeZoneRule * trsrules[],int32_t & trscount,UErrorCode & status)1192 SimpleTimeZone::getTimeZoneRules(const InitialTimeZoneRule*& initial,
1193                                  const TimeZoneRule* trsrules[],
1194                                  int32_t& trscount,
1195                                  UErrorCode& status) /*const*/ {
1196     if (U_FAILURE(status)) {
1197         return;
1198     }
1199     initTransitionRules(status);
1200     if (U_FAILURE(status)) {
1201         return;
1202     }
1203     initial = initialRule;
1204     int32_t cnt = 0;
1205     if (stdRule != NULL) {
1206         if (cnt < trscount) {
1207             trsrules[cnt++] = stdRule;
1208         }
1209         if (cnt < trscount) {
1210             trsrules[cnt++] = dstRule;
1211         }
1212     }
1213     trscount = cnt;
1214 }
1215 
1216 
1217 U_NAMESPACE_END
1218 
1219 #endif /* #if !UCONFIG_NO_FORMATTING */
1220 
1221 //eof
1222