]> git.saurik.com Git - apple/icu.git/blobdiff - icuSources/i18n/buddhcal.cpp
ICU-551.51.4.tar.gz
[apple/icu.git] / icuSources / i18n / buddhcal.cpp
index bf4290d07965cfce91c52046442fb7b39bb3d52d..99f2bd4c21ccbab31ecec0fe93392c2310e5954d 100644 (file)
@@ -1,6 +1,6 @@
 /*
 *******************************************************************************
-* Copyright (C) 2003, International Business Machines Corporation and    *
+* Copyright (C) 2003-2013, International Business Machines Corporation and    *
 * others. All Rights Reserved.                                                *
 *******************************************************************************
 *
 
 #include "buddhcal.h"
 #include "unicode/gregocal.h"
-#include "mutex.h"
+#include "umutex.h"
 #include <float.h>
 
 U_NAMESPACE_BEGIN
 
-const char BuddhistCalendar::fgClassID = 0; // Value is irrelevant
+UOBJECT_DEFINE_RTTI_IMPLEMENTATION(BuddhistCalendar)
 
-static const int32_t kMaxEra = 0; // only 1 era
+//static const int32_t kMaxEra = 0; // only 1 era
 
 static const int32_t kBuddhistEraStart = -543;  // 544 BC (Gregorian)
 
-static const int32_t kGregorianEpoch = 1970; 
+static const int32_t kGregorianEpoch = 1970;    // used as the default value of EXTENDED_YEAR
 
 BuddhistCalendar::BuddhistCalendar(const Locale& aLocale, UErrorCode& success)
-  :   GregorianCalendar(aLocale, success)
+:   GregorianCalendar(aLocale, success)
 {
     setTimeInMillis(getNow(), success); // Call this again now that the vtable is set up properly.
 }
@@ -41,217 +41,136 @@ BuddhistCalendar::~BuddhistCalendar()
 }
 
 BuddhistCalendar::BuddhistCalendar(const BuddhistCalendar& source)
-  : GregorianCalendar(source)
+: GregorianCalendar(source)
 {
 }
 
 BuddhistCalendar& BuddhistCalendar::operator= ( const BuddhistCalendar& right)
 {
-  GregorianCalendar::operator=(right);
-  return *this;
+    GregorianCalendar::operator=(right);
+    return *this;
 }
 
 Calendar* BuddhistCalendar::clone(void) const
 {
-  return new BuddhistCalendar(*this);
+    return new BuddhistCalendar(*this);
 }
 
 const char *BuddhistCalendar::getType() const
 {
-  return "buddhist";
+    return "buddhist";
 }
 
-int32_t
-BuddhistCalendar::getMaximum(UCalendarDateFields field) const
+int32_t BuddhistCalendar::handleGetExtendedYear()
 {
-  if(field == UCAL_ERA) {
-    return kMaxEra;
-  } else {
-    return GregorianCalendar::getMaximum(field);
-  }
-}
-
-int32_t
-BuddhistCalendar::getLeastMaximum(UCalendarDateFields field) const
-{
-  if(field == UCAL_ERA) {
-    return kMaxEra;
-  } else {
-    return GregorianCalendar::getLeastMaximum(field);
-  }
-}
-
-int32_t
-BuddhistCalendar::monthLength(int32_t month, int32_t year) const
-{
-  return GregorianCalendar::monthLength(month,year);
+    // EXTENDED_YEAR in BuddhistCalendar is a Gregorian year.
+    // The default value of EXTENDED_YEAR is 1970 (Buddhist 2513)
+    int32_t year;
+    if (newerField(UCAL_EXTENDED_YEAR, UCAL_YEAR) == UCAL_EXTENDED_YEAR) {
+        year = internalGet(UCAL_EXTENDED_YEAR, kGregorianEpoch);
+    } else {
+        // extended year is a gregorian year, where 1 = 1AD,  0 = 1BC, -1 = 2BC, etc 
+        year = internalGet(UCAL_YEAR, kGregorianEpoch - kBuddhistEraStart)
+                + kBuddhistEraStart;
+    }
+    return year;
 }
 
+int32_t BuddhistCalendar::handleComputeMonthStart(int32_t eyear, int32_t month,
 
-int32_t
-BuddhistCalendar::monthLength(int32_t month) const
+                                                  UBool useMonth) const
 {
-    UErrorCode status = U_ZERO_ERROR;
-    int32_t year = internalGet(UCAL_YEAR);
-    // ignore era
-    return GregorianCalendar::monthLength(month, getGregorianYear(status));
+    return GregorianCalendar::handleComputeMonthStart(eyear, month, useMonth);
 }
 
-int32_t BuddhistCalendar::internalGetEra() const
+void BuddhistCalendar::handleComputeFields(int32_t julianDay, UErrorCode& status)
 {
-    return isSet(UCAL_ERA) ? internalGet(UCAL_ERA) : BE;  
+    GregorianCalendar::handleComputeFields(julianDay, status);
+    int32_t y = internalGet(UCAL_EXTENDED_YEAR) - kBuddhistEraStart;
+    internalSet(UCAL_ERA, 0);
+    internalSet(UCAL_YEAR, y);
 }
 
-int32_t
-BuddhistCalendar::getGregorianYear(UErrorCode &status)  const
+int32_t BuddhistCalendar::handleGetLimit(UCalendarDateFields field, ELimitType limitType) const
 {
-  int32_t year = (fStamp[UCAL_YEAR] != kUnset) ? internalGet(UCAL_YEAR) : kGregorianEpoch+kBuddhistEraStart;
-  int32_t era = BE;
-  if (fStamp[UCAL_ERA] != kUnset) {
-    era = internalGet(UCAL_ERA);
-    if (era != BE) {
-      status = U_ILLEGAL_ARGUMENT_ERROR;
-      return kGregorianEpoch + kBuddhistEraStart;
+    if(field == UCAL_ERA) {
+        return BE;
+    } else {
+        return GregorianCalendar::handleGetLimit(field,limitType);
     }
-  }
-  return year + kBuddhistEraStart;
 }
 
+#if 0
 void BuddhistCalendar::timeToFields(UDate theTime, UBool quick, UErrorCode& status)
 {
-  GregorianCalendar::timeToFields(theTime, quick, status);
-
-  int32_t era = internalGet(UCAL_ERA);
-  int32_t year = internalGet(UCAL_YEAR);
-
-  if(era == GregorianCalendar::BC) {
-    year = 1-year;
-    era = BuddhistCalendar::BE;
-  } else if(era == GregorianCalendar::AD) {
-    era = BuddhistCalendar::BE;
-  } else {
-    status = U_INTERNAL_PROGRAM_ERROR;
-  }
-
-  year = year - kBuddhistEraStart;
-    
-  internalSet(UCAL_ERA, era);
-  internalSet(UCAL_YEAR, year);
-}
+    //Calendar::timeToFields(theTime, quick, status);
 
-void BuddhistCalendar::add(UCalendarDateFields field, int32_t amount, UErrorCode& status)
-{
-    if (U_FAILURE(status)) 
-        return;
-
-    if (amount == 0) 
-        return;   // Do nothing!
-    
-    if(field == UCAL_YEAR /* || field == UCAL_YEAR_WOY */) {
-        int32_t year = internalGet(field);
-        int32_t era = internalGetEra();
-
-        year += amount;
-        
-        set(field,year);
-        pinDayOfMonth();
+    int32_t era = internalGet(UCAL_ERA);
+    int32_t year = internalGet(UCAL_YEAR);
+
+    if(era == GregorianCalendar::BC) {
+        year = 1-year;
+        era = BuddhistCalendar::BE;
+    } else if(era == GregorianCalendar::AD) {
+        era = BuddhistCalendar::BE;
     } else {
-      GregorianCalendar::add(field,amount,status);
+        status = U_INTERNAL_PROGRAM_ERROR;
     }
-}
-
 
+    year = year - kBuddhistEraStart;
 
-// default century
-const UDate     BuddhistCalendar::fgSystemDefaultCentury        = DBL_MIN;
-const int32_t   BuddhistCalendar::fgSystemDefaultCenturyYear    = -1;
+    internalSet(UCAL_ERA, era);
+    internalSet(UCAL_YEAR, year);
+}
+#endif
 
-UDate           BuddhistCalendar::fgSystemDefaultCenturyStart       = DBL_MIN;
-int32_t         BuddhistCalendar::fgSystemDefaultCenturyStartYear   = -1;
+/**
+ * The system maintains a static default century start date.  This is initialized
+ * the first time it is used. Once the system default century date and year
+ * are set, they do not change.
+ */
+static UDate     gSystemDefaultCenturyStart       = DBL_MIN;
+static int32_t   gSystemDefaultCenturyStartYear   = -1;
+static icu::UInitOnce gBCInitOnce;
 
 
 UBool BuddhistCalendar::haveDefaultCentury() const
 {
-  return TRUE;
+    return TRUE;
 }
 
-UDate BuddhistCalendar::defaultCenturyStart() const
-{
-  return internalGetDefaultCenturyStart();
-}
-
-int32_t BuddhistCalendar::defaultCenturyStartYear() const
+static void U_CALLCONV
+initializeSystemDefaultCentury()
 {
-  return internalGetDefaultCenturyStartYear();
-}
-
-UDate
-BuddhistCalendar::internalGetDefaultCenturyStart() const
-{
-  // lazy-evaluate systemDefaultCenturyStart
-  UBool needsUpdate;
-  { 
-    Mutex m;
-    needsUpdate = (fgSystemDefaultCenturyStart == fgSystemDefaultCentury);
-  }
-
-  if (needsUpdate) {
-    initializeSystemDefaultCentury();
-  }
-
-  // use defaultCenturyStart unless it's the flag value;
-  // then use systemDefaultCenturyStart
-  
-  return fgSystemDefaultCenturyStart;
+    // initialize systemDefaultCentury and systemDefaultCenturyYear based
+    // on the current time.  They'll be set to 80 years before
+    // the current time.
+    UErrorCode status = U_ZERO_ERROR;
+    BuddhistCalendar calendar(Locale("@calendar=buddhist"),status);
+    if (U_SUCCESS(status)) {
+        calendar.setTime(Calendar::getNow(), status);
+        calendar.add(UCAL_YEAR, -80, status);
+        UDate    newStart =  calendar.getTime(status);
+        int32_t  newYear  =  calendar.get(UCAL_YEAR, status);
+        gSystemDefaultCenturyStartYear = newYear;
+        gSystemDefaultCenturyStart = newStart;
+    }
+    // We have no recourse upon failure unless we want to propagate the failure
+    // out.
 }
 
-int32_t
-BuddhistCalendar::internalGetDefaultCenturyStartYear() const
+UDate BuddhistCalendar::defaultCenturyStart() const
 {
-  // lazy-evaluate systemDefaultCenturyStartYear
-  UBool needsUpdate;
-  { 
-    Mutex m;
-    needsUpdate = (fgSystemDefaultCenturyStart == fgSystemDefaultCentury);
-  }
-
-  if (needsUpdate) {
-    initializeSystemDefaultCentury();
-  }
-
-  // use defaultCenturyStart unless it's the flag value;
-  // then use systemDefaultCenturyStartYear
-  
-  return    fgSystemDefaultCenturyStartYear;
+    // lazy-evaluate systemDefaultCenturyStart and systemDefaultCenturyStartYear
+    umtx_initOnce(gBCInitOnce, &initializeSystemDefaultCentury);
+    return gSystemDefaultCenturyStart;
 }
 
-void
-BuddhistCalendar::initializeSystemDefaultCentury()
+int32_t BuddhistCalendar::defaultCenturyStartYear() const
 {
-  // initialize systemDefaultCentury and systemDefaultCenturyYear based
-  // on the current time.  They'll be set to 80 years before
-  // the current time.
-  // No point in locking as it should be idempotent.
-  if (fgSystemDefaultCenturyStart == fgSystemDefaultCentury)
-  {
-    UErrorCode status = U_ZERO_ERROR;
-    Calendar *calendar = new BuddhistCalendar(Locale("th_TH_TRADITIONAL"),status);
-    if (calendar != NULL && U_SUCCESS(status))
-    {
-      calendar->setTime(Calendar::getNow(), status);
-      calendar->add(UCAL_YEAR, -80, status);
-      UDate    newStart =  calendar->getTime(status);
-      int32_t  newYear  =  calendar->get(UCAL_YEAR, status);
-      {
-        Mutex m;
-        fgSystemDefaultCenturyStart = newStart;
-        fgSystemDefaultCenturyStartYear = newYear;
-      }
-      delete calendar;
-    }
-    // We have no recourse upon failure unless we want to propagate the failure
-    // out.
-  }
+    // lazy-evaluate systemDefaultCenturyStartYear and systemDefaultCenturyStart 
+    umtx_initOnce(gBCInitOnce, &initializeSystemDefaultCentury);
+    return gSystemDefaultCenturyStartYear;
 }