mirror of
https://github.com/unicode-org/icu.git
synced 2025-04-06 22:15:31 +00:00
parent
a6efa924ad
commit
0b66fada30
24 changed files with 179 additions and 130 deletions
|
@ -3412,11 +3412,20 @@ int32_t Calendar::handleComputeJulianDay(UCalendarDateFields bestField, UErrorCo
|
|||
} else {
|
||||
dayOfMonth = getDefaultDayInMonth(year, month);
|
||||
}
|
||||
return julianDay + dayOfMonth;
|
||||
if (uprv_add32_overflow(dayOfMonth, julianDay, &dayOfMonth)) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return 0;
|
||||
}
|
||||
return dayOfMonth;
|
||||
}
|
||||
|
||||
if (bestField == UCAL_DAY_OF_YEAR) {
|
||||
return julianDay + internalGet(UCAL_DAY_OF_YEAR);
|
||||
int32_t result;
|
||||
if (uprv_add32_overflow(internalGet(UCAL_DAY_OF_YEAR), julianDay, &result)) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return 0;
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
int32_t firstDayOfWeek = getFirstDayOfWeek(); // Localized fdw
|
||||
|
|
|
@ -76,10 +76,27 @@ CECalendar::operator=(const CECalendar& right)
|
|||
// Calendar framework
|
||||
//-------------------------------------------------------------------------
|
||||
|
||||
int32_t
|
||||
int64_t
|
||||
CECalendar::handleComputeMonthStart(int32_t eyear,int32_t emonth, UBool /*useMonth*/) const
|
||||
{
|
||||
return ceToJD(eyear, emonth, 0, getJDEpochOffset());
|
||||
int64_t year64 = eyear;
|
||||
// handle month > 12, < 0 (e.g. from add/set)
|
||||
if ( emonth >= 0 ) {
|
||||
year64 += emonth/13;
|
||||
emonth %= 13;
|
||||
} else {
|
||||
++emonth;
|
||||
year64 += emonth/13 - 1;
|
||||
emonth = emonth%13 + 12;
|
||||
}
|
||||
|
||||
return (
|
||||
getJDEpochOffset() // difference from Julian epoch to 1,1,1
|
||||
+ 365LL * year64 // number of days from years
|
||||
+ ClockMath::floorDivide(year64, 4LL) // extra day of leap year
|
||||
+ 30 * emonth // number of days from months (months are 0-based)
|
||||
- 1 // number of days for present month (1 based)
|
||||
);
|
||||
}
|
||||
|
||||
int32_t
|
||||
|
@ -97,26 +114,6 @@ CECalendar::haveDefaultCentury() const
|
|||
//-------------------------------------------------------------------------
|
||||
// Calendar system Conversion methods...
|
||||
//-------------------------------------------------------------------------
|
||||
int32_t
|
||||
CECalendar::ceToJD(int32_t year, int32_t month, int32_t date, int32_t jdEpochOffset)
|
||||
{
|
||||
// handle month > 12, < 0 (e.g. from add/set)
|
||||
if ( month >= 0 ) {
|
||||
year += month/13;
|
||||
month %= 13;
|
||||
} else {
|
||||
++month;
|
||||
year += month/13 - 1;
|
||||
month = month%13 + 12;
|
||||
}
|
||||
return (int32_t) (
|
||||
jdEpochOffset // difference from Julian epoch to 1,1,1
|
||||
+ 365 * year // number of days from years
|
||||
+ ClockMath::floorDivide(year, 4) // extra day of leap year
|
||||
+ 30 * month // number of days from months (months are 0-based)
|
||||
+ date - 1 // number of days for present month (1 based)
|
||||
);
|
||||
}
|
||||
|
||||
void
|
||||
CECalendar::jdToCE(int32_t julianDay, int32_t jdEpochOffset, int32_t& year, int32_t& month, int32_t& day)
|
||||
|
|
|
@ -98,7 +98,7 @@ protected:
|
|||
* Return JD of start of given month/extended year
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
virtual int64_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
|
||||
/**
|
||||
* Calculate the limit for a specified type of limit and field
|
||||
|
@ -121,19 +121,6 @@ protected:
|
|||
*/
|
||||
virtual int32_t getJDEpochOffset() const = 0;
|
||||
|
||||
/**
|
||||
* Convert an Coptic/Ethiopic year, month, and day to a Julian day.
|
||||
*
|
||||
* @param year the extended year
|
||||
* @param month the month
|
||||
* @param day the day
|
||||
* @param jdEpochOffset the epoch offset from Julian epoch
|
||||
* @return Julian day
|
||||
* @internal
|
||||
*/
|
||||
static int32_t ceToJD(int32_t year, int32_t month, int32_t date,
|
||||
int32_t jdEpochOffset);
|
||||
|
||||
/**
|
||||
* Convert a Julian day to an Coptic/Ethiopic year, month and day
|
||||
*
|
||||
|
|
|
@ -324,7 +324,7 @@ const UFieldResolutionTable* ChineseCalendar::getFieldResolutionTable() const {
|
|||
* day of the given month and year
|
||||
* @stable ICU 2.8
|
||||
*/
|
||||
int32_t ChineseCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const {
|
||||
int64_t ChineseCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const {
|
||||
// If the month is out of range, adjust it into range, and
|
||||
// modify the extended year value accordingly.
|
||||
if (month < 0 || month > 11) {
|
||||
|
@ -337,7 +337,7 @@ int32_t ChineseCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, U
|
|||
int32_t theNewYear = newYear(gyear);
|
||||
int32_t newMoon = newMoonNear(theNewYear + month * 29, true);
|
||||
|
||||
int32_t julianDay = newMoon + kEpochStartAsJulianDay;
|
||||
int64_t julianDay = newMoon + kEpochStartAsJulianDay;
|
||||
|
||||
// Ignore IS_LEAP_MONTH field if useMonth is false
|
||||
int32_t isLeapMonth = useMonth ? internalGet(UCAL_IS_LEAP_MONTH) : 0;
|
||||
|
@ -546,7 +546,12 @@ int32_t ChineseCalendar::winterSolstice(int32_t gyear) const {
|
|||
umtx_unlock(&astroLock);
|
||||
|
||||
// Winter solstice is 270 degrees solar longitude aka Dongzhi
|
||||
cacheValue = (int32_t)millisToDays(solarLong);
|
||||
double days = millisToDays(solarLong);
|
||||
if (days < INT32_MIN || days > INT32_MAX) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return 0;
|
||||
}
|
||||
cacheValue = (int32_t) days;
|
||||
CalendarCache::put(&gChineseCalendarWinterSolsticeCache, gyear, cacheValue, status);
|
||||
}
|
||||
if(U_FAILURE(status)) {
|
||||
|
|
|
@ -208,7 +208,7 @@ class U_I18N_API ChineseCalendar : public Calendar {
|
|||
protected:
|
||||
virtual int32_t handleGetLimit(UCalendarDateFields field, ELimitType limitType) const override;
|
||||
virtual int32_t handleGetMonthLength(int32_t extendedYear, int32_t month) const override;
|
||||
virtual int32_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
virtual int64_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
virtual int32_t handleGetExtendedYear() override;
|
||||
virtual void handleComputeFields(int32_t julianDay, UErrorCode &status) override;
|
||||
virtual const UFieldResolutionTable* getFieldResolutionTable() const override;
|
||||
|
|
|
@ -161,21 +161,6 @@ CopticCalendar::getJDEpochOffset() const
|
|||
}
|
||||
|
||||
|
||||
#if 0
|
||||
// We do not want to introduce this API in ICU4C.
|
||||
// It was accidentally introduced in ICU4J as a public API.
|
||||
|
||||
//-------------------------------------------------------------------------
|
||||
// Calendar system Conversion methods...
|
||||
//-------------------------------------------------------------------------
|
||||
|
||||
int32_t
|
||||
CopticCalendar::copticToJD(int32_t year, int32_t month, int32_t day)
|
||||
{
|
||||
return CECalendar::ceToJD(year, month, day, COPTIC_JD_EPOCH_OFFSET);
|
||||
}
|
||||
#endif
|
||||
|
||||
U_NAMESPACE_END
|
||||
|
||||
#endif /* #if !UCONFIG_NO_FORMATTING */
|
||||
|
|
|
@ -150,21 +150,6 @@ EthiopicCalendar::getJDEpochOffset() const
|
|||
}
|
||||
|
||||
|
||||
#if 0
|
||||
// We do not want to introduce this API in ICU4C.
|
||||
// It was accidentally introduced in ICU4J as a public API.
|
||||
|
||||
//-------------------------------------------------------------------------
|
||||
// Calendar system Conversion methods...
|
||||
//-------------------------------------------------------------------------
|
||||
|
||||
int32_t
|
||||
EthiopicCalendar::ethiopicToJD(int32_t year, int32_t month, int32_t date)
|
||||
{
|
||||
return ceToJD(year, month, date, JD_EPOCH_OFFSET_AMETE_MIHRET);
|
||||
}
|
||||
#endif
|
||||
|
||||
//-------------------------------------------------------------------------
|
||||
// Constructors...
|
||||
//-------------------------------------------------------------------------
|
||||
|
|
|
@ -536,7 +536,7 @@ int32_t GregorianCalendar::handleComputeJulianDay(UCalendarDateFields bestField,
|
|||
return jd;
|
||||
}
|
||||
|
||||
int32_t GregorianCalendar::handleComputeMonthStart(int32_t eyear, int32_t month,
|
||||
int64_t GregorianCalendar::handleComputeMonthStart(int32_t eyear, int32_t month,
|
||||
|
||||
UBool /* useMonth */) const
|
||||
{
|
||||
|
@ -580,7 +580,7 @@ int32_t GregorianCalendar::handleComputeMonthStart(int32_t eyear, int32_t month,
|
|||
julianDay += isLeap?kLeapNumDays[month]:kNumDays[month];
|
||||
}
|
||||
|
||||
return static_cast<int32_t>(julianDay);
|
||||
return julianDay;
|
||||
}
|
||||
|
||||
int32_t GregorianCalendar::handleGetMonthLength(int32_t extendedYear, int32_t month) const
|
||||
|
|
|
@ -41,7 +41,7 @@ int32_t ClockMath::floorDivide(int32_t numerator, int32_t denominator,
|
|||
return quotient;
|
||||
}
|
||||
|
||||
int32_t ClockMath::floorDivide(double numerator, int32_t denominator,
|
||||
double ClockMath::floorDivide(double numerator, int32_t denominator,
|
||||
int32_t* remainder) {
|
||||
// For an integer n and representable ⌊x/n⌋, ⌊RN(x/n)⌋=⌊x/n⌋, where RN is
|
||||
// rounding to nearest.
|
||||
|
@ -53,7 +53,7 @@ int32_t ClockMath::floorDivide(double numerator, int32_t denominator,
|
|||
// n+⌈x⌉ = ⌊x+n⌋ + 1. Rewriting it as ⌊x⌋+n makes the addition exact.
|
||||
*remainder = (int32_t) (uprv_floor(numerator) - (quotient * denominator));
|
||||
}
|
||||
return (int32_t) quotient;
|
||||
return quotient;
|
||||
}
|
||||
|
||||
double ClockMath::floorDivide(double dividend, double divisor,
|
||||
|
@ -104,12 +104,12 @@ const int8_t Grego::MONTH_LENGTH[24] =
|
|||
{31,28,31,30,31,30,31,31,30,31,30,31,
|
||||
31,29,31,30,31,30,31,31,30,31,30,31};
|
||||
|
||||
double Grego::fieldsToDay(int32_t year, int32_t month, int32_t dom) {
|
||||
int64_t Grego::fieldsToDay(int32_t year, int32_t month, int32_t dom) {
|
||||
|
||||
int32_t y = year - 1;
|
||||
int64_t y = year - 1;
|
||||
|
||||
double julian = 365 * y + ClockMath::floorDivide(y, 4) + (JULIAN_1_CE - 3) + // Julian cal
|
||||
ClockMath::floorDivide(y, 400) - ClockMath::floorDivide(y, 100) + 2 + // => Gregorian cal
|
||||
int64_t julian = 365LL * y + ClockMath::floorDivide(y, 4LL) + (JULIAN_1_CE - 3) + // Julian cal
|
||||
ClockMath::floorDivide(y, 400LL) - ClockMath::floorDivide(y, 100LL) + 2 + // => Gregorian cal
|
||||
DAYS_BEFORE[month + (isLeapYear(year) ? 12 : 0)] + dom; // => month/dom
|
||||
|
||||
return julian - JULIAN_1970_CE; // JD => epoch day
|
||||
|
|
|
@ -95,7 +95,7 @@ class ClockMath {
|
|||
* |denominator|)</code>.
|
||||
* @return the floor of the quotient
|
||||
*/
|
||||
static int32_t floorDivide(double numerator, int32_t denominator,
|
||||
static double floorDivide(double numerator, int32_t denominator,
|
||||
int32_t* remainder);
|
||||
|
||||
/**
|
||||
|
@ -196,7 +196,7 @@ class Grego {
|
|||
* @param dom 1-based day of month
|
||||
* @return the day number, with day 0 == Jan 1 1970
|
||||
*/
|
||||
static double fieldsToDay(int32_t year, int32_t month, int32_t dom);
|
||||
static int64_t fieldsToDay(int32_t year, int32_t month, int32_t dom);
|
||||
|
||||
/**
|
||||
* Convert a 1970-epoch day number to proleptic Gregorian year,
|
||||
|
|
|
@ -394,14 +394,14 @@ static const int32_t BAHARAD = 11*HOUR_PARTS + 204;
|
|||
int32_t HebrewCalendar::startOfYear(int32_t year, UErrorCode &status)
|
||||
{
|
||||
ucln_i18n_registerCleanup(UCLN_I18N_HEBREW_CALENDAR, calendar_hebrew_cleanup);
|
||||
int32_t day = CalendarCache::get(&gCache, year, status);
|
||||
int64_t day = CalendarCache::get(&gCache, year, status);
|
||||
|
||||
if (day == 0) {
|
||||
// # of months before year
|
||||
int32_t months = (int32_t)ClockMath::floorDivide((235 * (int64_t)year - 234), (int64_t)19);
|
||||
int64_t months = ClockMath::floorDivide((235 * (int64_t)year - 234), (int64_t)19);
|
||||
|
||||
int64_t frac = (int64_t)months * MONTH_FRACT + BAHARAD; // Fractional part of day #
|
||||
day = months * 29 + (int32_t)(frac / DAY_PARTS); // Whole # part of calculation
|
||||
int64_t frac = months * MONTH_FRACT + BAHARAD; // Fractional part of day #
|
||||
day = months * 29LL + frac / DAY_PARTS; // Whole # part of calculation
|
||||
frac = frac % DAY_PARTS; // Time of day
|
||||
|
||||
int32_t wd = (day % 7); // Day of week (0 == Monday)
|
||||
|
@ -423,8 +423,14 @@ int32_t HebrewCalendar::startOfYear(int32_t year, UErrorCode &status)
|
|||
// Prevents 382-day years.
|
||||
day += 1;
|
||||
}
|
||||
CalendarCache::put(&gCache, year, day, status);
|
||||
if (day > INT32_MAX || day < INT32_MIN) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return 0;
|
||||
}
|
||||
CalendarCache::put(&gCache, year, static_cast<int32_t>(day), status);
|
||||
}
|
||||
// Out of range value is alread rejected before putting into cache.
|
||||
U_ASSERT(INT32_MIN <= day && day <= INT32_MAX);
|
||||
return day;
|
||||
}
|
||||
|
||||
|
@ -478,7 +484,7 @@ int32_t HebrewCalendar::yearType(int32_t year) const
|
|||
*/
|
||||
UBool HebrewCalendar::isLeapYear(int32_t year) {
|
||||
//return (year * 12 + 17) % 19 >= 12;
|
||||
int32_t x = (year*12 + 17) % 19;
|
||||
int64_t x = (year*12LL + 17) % 19;
|
||||
return x >= ((x < 0) ? -7 : 12);
|
||||
}
|
||||
|
||||
|
@ -643,7 +649,7 @@ int32_t HebrewCalendar::handleGetExtendedYear() {
|
|||
* Return JD of start of given month/year.
|
||||
* @internal
|
||||
*/
|
||||
int32_t HebrewCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool /*useMonth*/) const {
|
||||
int64_t HebrewCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool /*useMonth*/) const {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
// Resolve out-of-range months. This is necessary in order to
|
||||
// obtain the correct year. We correct to
|
||||
|
@ -659,7 +665,7 @@ int32_t HebrewCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UB
|
|||
month -= monthsInYear(eyear++);
|
||||
}
|
||||
|
||||
int32_t day = startOfYear(eyear, status);
|
||||
int64_t day = startOfYear(eyear, status);
|
||||
|
||||
if(U_FAILURE(status)) {
|
||||
return 0;
|
||||
|
@ -673,7 +679,7 @@ int32_t HebrewCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UB
|
|||
}
|
||||
}
|
||||
|
||||
return (int) (day + 347997);
|
||||
return day + 347997LL;
|
||||
}
|
||||
|
||||
constexpr uint32_t kHebrewRelatedYearDiff = -3760;
|
||||
|
|
|
@ -367,7 +367,7 @@ public:
|
|||
* day of the given month and year
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t handleComputeMonthStart(int32_t eyear, int32_t month,
|
||||
virtual int64_t handleComputeMonthStart(int32_t eyear, int32_t month,
|
||||
UBool useMonth) const override;
|
||||
|
||||
|
||||
|
|
|
@ -203,7 +203,7 @@ static double IndianToJD(int32_t year, int32_t month, int32_t date) {
|
|||
* @param eyear The year in Indian Calendar measured from Saka Era (78 AD).
|
||||
* @param month The month in Indian calendar
|
||||
*/
|
||||
int32_t IndianCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool /* useMonth */ ) const {
|
||||
int64_t IndianCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool /* useMonth */ ) const {
|
||||
|
||||
//month is 0 based; converting it to 1-based
|
||||
int32_t imonth;
|
||||
|
@ -219,9 +219,9 @@ int32_t IndianCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UB
|
|||
imonth = month + 1;
|
||||
}
|
||||
|
||||
double jd = IndianToJD(eyear ,imonth, 1);
|
||||
int64_t jd = IndianToJD(eyear ,imonth, 1);
|
||||
|
||||
return (int32_t)jd;
|
||||
return jd;
|
||||
}
|
||||
|
||||
//-------------------------------------------------------------------------
|
||||
|
|
|
@ -225,7 +225,7 @@ public:
|
|||
/**
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
virtual int64_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
|
||||
//-------------------------------------------------------------------------
|
||||
// Functions for converting from milliseconds to field values
|
||||
|
|
|
@ -320,7 +320,7 @@ UBool IslamicCalendar::civilLeapYear(int32_t year)
|
|||
* Return the day # on which the given year starts. Days are counted
|
||||
* from the Hijri epoch, origin 0.
|
||||
*/
|
||||
int32_t IslamicCalendar::yearStart(int32_t year) const{
|
||||
int64_t IslamicCalendar::yearStart(int32_t year) const{
|
||||
return trueMonthStart(12*(year-1));
|
||||
}
|
||||
|
||||
|
@ -331,7 +331,7 @@ int32_t IslamicCalendar::yearStart(int32_t year) const{
|
|||
* @param year The hijri year
|
||||
* @param month The hijri month, 0-based (assumed to be in range 0..11)
|
||||
*/
|
||||
int32_t IslamicCalendar::monthStart(int32_t year, int32_t month) const {
|
||||
int64_t IslamicCalendar::monthStart(int32_t year, int32_t month) const {
|
||||
return trueMonthStart(12*(year-1) + month);
|
||||
}
|
||||
|
||||
|
@ -463,7 +463,7 @@ int32_t IslamicCalendar::handleGetYearLength(int32_t extendedYear) const {
|
|||
/**
|
||||
* @draft ICU 2.4
|
||||
*/
|
||||
int32_t IslamicCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool /* useMonth */) const {
|
||||
int64_t IslamicCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool /* useMonth */) const {
|
||||
// This may be called by Calendar::handleComputeJulianDay with months out of the range
|
||||
// 0..11. Need to handle that here since monthStart requires months in the range 0.11.
|
||||
if (month > 11) {
|
||||
|
@ -538,10 +538,18 @@ void IslamicCalendar::handleComputeFields(int32_t julianDay, UErrorCode &status)
|
|||
|
||||
int32_t year = month >= 0 ? ((month / 12) + 1) : ((month + 1 ) / 12);
|
||||
month = ((month % 12) + 12 ) % 12;
|
||||
int32_t dayOfMonth = (days - monthStart(year, month)) + 1;
|
||||
int64_t dayOfMonth = (days - monthStart(year, month)) + 1;
|
||||
if (dayOfMonth > INT32_MAX || dayOfMonth < INT32_MIN) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return;
|
||||
}
|
||||
|
||||
// Now figure out the day of the year.
|
||||
int32_t dayOfYear = (days - monthStart(year, 0)) + 1;
|
||||
int64_t dayOfYear = (days - monthStart(year, 0)) + 1;
|
||||
if (dayOfYear > INT32_MAX || dayOfYear < INT32_MIN) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return;
|
||||
}
|
||||
|
||||
internalSet(UCAL_ERA, 0);
|
||||
internalSet(UCAL_YEAR, year);
|
||||
|
@ -685,9 +693,9 @@ IslamicCivilCalendar* IslamicCivilCalendar::clone() const {
|
|||
* Return the day # on which the given year starts. Days are counted
|
||||
* from the Hijri epoch, origin 0.
|
||||
*/
|
||||
int32_t IslamicCivilCalendar::yearStart(int32_t year) const{
|
||||
return static_cast<int32_t>(
|
||||
(year-1)*354 + ClockMath::floorDivide((3+11*static_cast<int64_t>(year)),
|
||||
int64_t IslamicCivilCalendar::yearStart(int32_t year) const{
|
||||
return static_cast<int64_t>(
|
||||
(year-1)*354LL + ClockMath::floorDivide((3+11*static_cast<int64_t>(year)),
|
||||
static_cast<int64_t>(30)));
|
||||
}
|
||||
|
||||
|
@ -698,10 +706,10 @@ int32_t IslamicCivilCalendar::yearStart(int32_t year) const{
|
|||
* @param year The hijri year
|
||||
* @param month The hijri month, 0-based (assumed to be in range 0..11)
|
||||
*/
|
||||
int32_t IslamicCivilCalendar::monthStart(int32_t year, int32_t month) const {
|
||||
int64_t IslamicCivilCalendar::monthStart(int32_t year, int32_t month) const {
|
||||
// This does not handle months out of the range 0..11
|
||||
return static_cast<int32_t>(
|
||||
uprv_ceil(29.5*month) + (year-1)*354 +
|
||||
return static_cast<int64_t>(
|
||||
uprv_ceil(29.5*month) + (year-1)*354LL +
|
||||
static_cast<int32_t>(ClockMath::floorDivide(
|
||||
3+11*static_cast<int64_t>(year),
|
||||
static_cast<int64_t>(30))));
|
||||
|
@ -758,10 +766,18 @@ void IslamicCivilCalendar::handleComputeFields(int32_t julianDay, UErrorCode &st
|
|||
uprv_ceil((days - 29 - yearStart(year)) / 29.5 ));
|
||||
month = month<11?month:11;
|
||||
|
||||
int32_t dayOfMonth = (days - monthStart(year, month)) + 1;
|
||||
int64_t dayOfMonth = (days - monthStart(year, month)) + 1;
|
||||
if (dayOfMonth > INT32_MAX || dayOfMonth < INT32_MIN) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return;
|
||||
}
|
||||
|
||||
// Now figure out the day of the year.
|
||||
int32_t dayOfYear = (days - monthStart(year, 0)) + 1;
|
||||
int64_t dayOfYear = (days - monthStart(year, 0)) + 1;
|
||||
if (dayOfYear > INT32_MAX || dayOfYear < INT32_MIN) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return;
|
||||
}
|
||||
|
||||
internalSet(UCAL_ERA, 0);
|
||||
internalSet(UCAL_YEAR, year);
|
||||
|
@ -819,15 +835,15 @@ IslamicUmalquraCalendar* IslamicUmalquraCalendar::clone() const {
|
|||
* Return the day # on which the given year starts. Days are counted
|
||||
* from the Hijri epoch, origin 0.
|
||||
*/
|
||||
int32_t IslamicUmalquraCalendar::yearStart(int32_t year) const {
|
||||
int64_t IslamicUmalquraCalendar::yearStart(int32_t year) const {
|
||||
if (year < UMALQURA_YEAR_START || year > UMALQURA_YEAR_END) {
|
||||
return static_cast<int32_t>(
|
||||
(year-1)*354 + ClockMath::floorDivide((3+11*static_cast<int64_t>(year)),
|
||||
return static_cast<int64_t>(
|
||||
(year-1)*354LL + ClockMath::floorDivide((3+11*static_cast<int64_t>(year)),
|
||||
static_cast<int64_t>(30)));
|
||||
}
|
||||
year -= UMALQURA_YEAR_START;
|
||||
// rounded least-squares fit of the dates previously calculated from UMALQURA_MONTHLENGTH iteration
|
||||
int32_t yrStartLinearEstimate = static_cast<int32_t>(
|
||||
int64_t yrStartLinearEstimate = static_cast<int64_t>(
|
||||
(354.36720 * (double)year) + 460322.05 + 0.5);
|
||||
// need a slight correction to some
|
||||
return yrStartLinearEstimate + umAlQuraYrStartEstimateFix[year];
|
||||
|
@ -840,8 +856,8 @@ int32_t IslamicUmalquraCalendar::yearStart(int32_t year) const {
|
|||
* @param year The hijri year
|
||||
* @param month The hijri month, 0-based (assumed to be in range 0..11)
|
||||
*/
|
||||
int32_t IslamicUmalquraCalendar::monthStart(int32_t year, int32_t month) const {
|
||||
int32_t ms = yearStart(year);
|
||||
int64_t IslamicUmalquraCalendar::monthStart(int32_t year, int32_t month) const {
|
||||
int64_t ms = yearStart(year);
|
||||
for(int i=0; i< month; i++){
|
||||
ms+= handleGetMonthLength(year, i);
|
||||
}
|
||||
|
@ -899,10 +915,11 @@ int32_t IslamicUmalquraCalendar::handleGetYearLength(int32_t extendedYear) const
|
|||
*/
|
||||
void IslamicUmalquraCalendar::handleComputeFields(int32_t julianDay, UErrorCode &status) {
|
||||
if (U_FAILURE(status)) return;
|
||||
int32_t year, month, dayOfMonth, dayOfYear;
|
||||
int32_t year, month, dayOfYear;
|
||||
int64_t dayOfMonth;
|
||||
int32_t days = julianDay - getEpoc();
|
||||
|
||||
int32_t umalquraStartdays = yearStart(UMALQURA_YEAR_START) ;
|
||||
int64_t umalquraStartdays = yearStart(UMALQURA_YEAR_START) ;
|
||||
if (days < umalquraStartdays) {
|
||||
//Use Civil calculation
|
||||
year = (int32_t)ClockMath::floorDivide(
|
||||
|
@ -935,9 +952,17 @@ void IslamicUmalquraCalendar::handleComputeFields(int32_t julianDay, UErrorCode
|
|||
}
|
||||
|
||||
dayOfMonth = (days - monthStart(year, month)) + 1;
|
||||
if (dayOfMonth > INT32_MAX || dayOfMonth < INT32_MIN) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return;
|
||||
}
|
||||
|
||||
// Now figure out the day of the year.
|
||||
dayOfYear = (days - monthStart(year, 0)) + 1;
|
||||
if (dayOfYear > INT32_MAX || dayOfYear < INT32_MIN) {
|
||||
status = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return;
|
||||
}
|
||||
|
||||
internalSet(UCAL_ERA, 0);
|
||||
internalSet(UCAL_YEAR, year);
|
||||
|
|
|
@ -206,7 +206,7 @@ class U_I18N_API IslamicCalendar : public Calendar {
|
|||
* Return the day # on which the given year starts. Days are counted
|
||||
* from the Hijri epoch, origin 0.
|
||||
*/
|
||||
virtual int32_t yearStart(int32_t year) const;
|
||||
virtual int64_t yearStart(int32_t year) const;
|
||||
|
||||
/**
|
||||
* Return the day # on which the given month starts. Days are counted
|
||||
|
@ -215,7 +215,7 @@ class U_I18N_API IslamicCalendar : public Calendar {
|
|||
* @param year The hijri year
|
||||
* @param year The hijri month, 0-based
|
||||
*/
|
||||
virtual int32_t monthStart(int32_t year, int32_t month) const;
|
||||
virtual int64_t monthStart(int32_t year, int32_t month) const;
|
||||
|
||||
/**
|
||||
* Find the day number on which a particular month of the true/lunar
|
||||
|
@ -271,7 +271,7 @@ class U_I18N_API IslamicCalendar : public Calendar {
|
|||
/**
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
virtual int64_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
|
||||
//-------------------------------------------------------------------------
|
||||
// Functions for converting from milliseconds to field values
|
||||
|
@ -463,7 +463,7 @@ class U_I18N_API IslamicCivilCalendar : public IslamicCalendar {
|
|||
* from the Hijri epoch, origin 0.
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t yearStart(int32_t year) const override;
|
||||
virtual int64_t yearStart(int32_t year) const override;
|
||||
|
||||
/**
|
||||
* Return the day # on which the given month starts. Days are counted
|
||||
|
@ -473,7 +473,7 @@ class U_I18N_API IslamicCivilCalendar : public IslamicCalendar {
|
|||
* @param year The hijri month, 0-based
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t monthStart(int32_t year, int32_t month) const override;
|
||||
virtual int64_t monthStart(int32_t year, int32_t month) const override;
|
||||
|
||||
/**
|
||||
* Return the length (in days) of the given month.
|
||||
|
@ -646,7 +646,7 @@ class U_I18N_API IslamicUmalquraCalendar : public IslamicCalendar {
|
|||
* from the Hijri epoch, origin 0.
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t yearStart(int32_t year) const override;
|
||||
virtual int64_t yearStart(int32_t year) const override;
|
||||
|
||||
/**
|
||||
* Return the day # on which the given month starts. Days are counted
|
||||
|
@ -656,7 +656,7 @@ class U_I18N_API IslamicUmalquraCalendar : public IslamicCalendar {
|
|||
* @param year The hijri month, 0-based
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t monthStart(int32_t year, int32_t month) const override;
|
||||
virtual int64_t monthStart(int32_t year, int32_t month) const override;
|
||||
|
||||
/**
|
||||
* Return the length (in days) of the given month.
|
||||
|
|
|
@ -166,14 +166,14 @@ int32_t PersianCalendar::handleGetYearLength(int32_t extendedYear) const {
|
|||
//-------------------------------------------------------------------------
|
||||
|
||||
// Return JD of start of given month/year
|
||||
int32_t PersianCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool /*useMonth*/) const {
|
||||
int64_t PersianCalendar::handleComputeMonthStart(int32_t eyear, int32_t month, UBool /*useMonth*/) const {
|
||||
// If the month is out of range, adjust it into range, and
|
||||
// modify the extended year value accordingly.
|
||||
if (month < 0 || month > 11) {
|
||||
eyear += ClockMath::floorDivide(month, 12, &month);
|
||||
}
|
||||
|
||||
int32_t julianDay = PERSIAN_EPOCH - 1 + 365 * (eyear - 1) + ClockMath::floorDivide(8 * eyear + 21, 33);
|
||||
int64_t julianDay = PERSIAN_EPOCH - 1 + 365LL * (eyear - 1) + ClockMath::floorDivide(8 * eyear + 21, 33);
|
||||
|
||||
if (month != 0) {
|
||||
julianDay += kPersianNumDays[month];
|
||||
|
|
|
@ -219,7 +219,7 @@ class PersianCalendar : public Calendar {
|
|||
/**
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
virtual int64_t handleComputeMonthStart(int32_t eyear, int32_t month, UBool useMonth) const override;
|
||||
|
||||
//-------------------------------------------------------------------------
|
||||
// Functions for converting from milliseconds to field values
|
||||
|
|
|
@ -733,6 +733,11 @@ void TimeZone::getOffset(UDate date, UBool local, int32_t& rawOffset,
|
|||
int32_t year, month, dom, dow, millis;
|
||||
double day = ClockMath::floorDivide(date, U_MILLIS_PER_DAY, &millis);
|
||||
|
||||
// out of the range
|
||||
if (day < INT32_MIN || day > INT32_MAX) {
|
||||
ec = U_ILLEGAL_ARGUMENT_ERROR;
|
||||
return;
|
||||
}
|
||||
Grego::dayToFields(day, year, month, dom, dow);
|
||||
|
||||
dstOffset = getOffset(GregorianCalendar::AD, year, month, dom,
|
||||
|
|
|
@ -1667,7 +1667,7 @@ protected:
|
|||
* day of the given month and year
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t handleComputeMonthStart(int32_t eyear, int32_t month,
|
||||
virtual int64_t handleComputeMonthStart(int32_t eyear, int32_t month,
|
||||
UBool useMonth) const = 0;
|
||||
|
||||
/**
|
||||
|
|
|
@ -487,7 +487,7 @@ public:
|
|||
* day of the given month and year
|
||||
* @internal
|
||||
*/
|
||||
virtual int32_t handleComputeMonthStart(int32_t eyear, int32_t month,
|
||||
virtual int64_t handleComputeMonthStart(int32_t eyear, int32_t month,
|
||||
UBool useMonth) const override;
|
||||
|
||||
/**
|
||||
|
|
|
@ -79,6 +79,7 @@ extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
|
|||
if (U_FAILURE(status)) return 0;
|
||||
std::unique_ptr<icu::Calendar> cal(
|
||||
icu::Calendar::createInstance(*timeZone, locale, status));
|
||||
printf("locale = %s\n", locale.getName());
|
||||
if (U_FAILURE(status)) return 0;
|
||||
cal->clear();
|
||||
|
||||
|
|
|
@ -190,8 +190,15 @@ void CalendarTest::runIndexedTest( int32_t index, UBool exec, const char* &name,
|
|||
TESTCASE_AUTO(TestDangiOverflowIsLeapMonthBetween22507);
|
||||
TESTCASE_AUTO(TestRollWeekOfYear);
|
||||
TESTCASE_AUTO(TestFirstDayOfWeek);
|
||||
|
||||
TESTCASE_AUTO(Test22633ChineseOverflow);
|
||||
TESTCASE_AUTO(Test22633IndianOverflow);
|
||||
TESTCASE_AUTO(Test22633IslamicUmalquraOverflow);
|
||||
TESTCASE_AUTO(Test22633PersianOverflow);
|
||||
|
||||
TESTCASE_AUTO(TestAddOverflow);
|
||||
|
||||
|
||||
TESTCASE_AUTO(TestChineseCalendarComputeMonthStart);
|
||||
|
||||
TESTCASE_AUTO_END;
|
||||
|
@ -5632,12 +5639,45 @@ void CalendarTest::TestFirstDayOfWeek() {
|
|||
verifyFirstDayOfWeek("zxx", UCAL_MONDAY);
|
||||
}
|
||||
|
||||
void CalendarTest::Test22633ChineseOverflow() {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=chinese"), status), status);
|
||||
U_ASSERT(U_SUCCESS(status));
|
||||
cal->setTime(2043071457431218011677338081118001787485161156097100985923226601036925437809699842362992455895409920480414647512899096575018732258582416938813614617757317338664031880042592085084690242819214720523061081124318514531466365480449329351434046537728.000000, status);
|
||||
U_ASSERT(U_SUCCESS(status));
|
||||
cal->set(UCAL_EXTENDED_YEAR, -1594662558);
|
||||
cal->get(UCAL_YEAR, status);
|
||||
assertTrue("Should return success", U_SUCCESS(status));
|
||||
}
|
||||
void CalendarTest::Test22633IndianOverflow() {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=indian"), status), status);
|
||||
U_ASSERT(U_SUCCESS(status));
|
||||
cal->roll(UCAL_EXTENDED_YEAR, -2120158417, status);
|
||||
assertTrue("Should return success", U_SUCCESS(status));
|
||||
}
|
||||
void CalendarTest::Test22633IslamicUmalquraOverflow() {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=islamic-umalqura"), status), status);
|
||||
U_ASSERT(U_SUCCESS(status));
|
||||
cal->roll(UCAL_YEAR, -134404585, status);
|
||||
assertTrue("Should return success", U_SUCCESS(status));
|
||||
}
|
||||
|
||||
void CalendarTest::Test22633PersianOverflow() {
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=persian"), status), status);
|
||||
U_ASSERT(U_SUCCESS(status));
|
||||
cal->add(UCAL_ORDINAL_MONTH, 1594095615, status);
|
||||
assertTrue("Should return success", U_SUCCESS(status));
|
||||
}
|
||||
|
||||
void CalendarTest::TestChineseCalendarComputeMonthStart() { // ICU-22639
|
||||
UErrorCode status = U_ZERO_ERROR;
|
||||
|
||||
// An extended year for which hasLeapMonthBetweenWinterSolstices is true.
|
||||
constexpr int32_t eyear = 4643;
|
||||
constexpr int32_t monthStart = 2453764;
|
||||
constexpr int64_t monthStart = 2453764;
|
||||
|
||||
LocalPointer<Calendar> calendar(
|
||||
Calendar::createInstance(Locale("en_US@calendar=chinese"), status),
|
||||
|
|
|
@ -334,6 +334,10 @@ public: // package
|
|||
void TestFWWithISO8601();
|
||||
void TestAddOverflow();
|
||||
void TestRollWeekOfYear();
|
||||
void Test22633ChineseOverflow();
|
||||
void Test22633IndianOverflow();
|
||||
void Test22633IslamicUmalquraOverflow();
|
||||
void Test22633PersianOverflow();
|
||||
|
||||
void verifyFirstDayOfWeek(const char* locale, UCalendarDaysOfWeek expected);
|
||||
void TestFirstDayOfWeek();
|
||||
|
|
Loading…
Add table
Reference in a new issue