⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 datemath.cpp

📁 linux下开源浏览器WebKit的源码,市面上的很多商用浏览器都是移植自WebKit
💻 CPP
📖 第 1 页 / 共 2 页
字号:
/* * Copyright (C) 1999-2000 Harri Porten (porten@kde.org) * Copyright (C) 2006, 2007 Apple Inc. All rights reserved. *  * The Original Code is Mozilla Communicator client code, released * March 31, 1998. * * The Initial Developer of the Original Code is * Netscape Communications Corporation. * Portions created by the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA * * Alternatively, the contents of this file may be used under the terms * of either the Mozilla Public License Version 1.1, found at * http://www.mozilla.org/MPL/ (the "MPL") or the GNU General Public * License Version 2.0, found at http://www.fsf.org/copyleft/gpl.html * (the "GPL"), in which case the provisions of the MPL or the GPL are * applicable instead of those above.  If you wish to allow use of your * version of this file only under the terms of one of those two * licenses (the MPL or the GPL) and not to allow others to use your * version of this file under the LGPL, indicate your decision by * deletingthe provisions above and replace them with the notice and * other provisions required by the MPL or the GPL, as the case may be. * If you do not delete the provisions above, a recipient may use your * version of this file under any of the LGPL, the MPL or the GPL. */#include "config.h"#include "DateMath.h"#include "JSNumberCell.h"#include <math.h>#include <stdint.h>#include <time.h>#include <wtf/ASCIICType.h>#include <wtf/Assertions.h>#include <wtf/CurrentTime.h>#include <wtf/MathExtras.h>#include <wtf/StringExtras.h>#if HAVE(ERRNO_H)#include <errno.h>#endif#if PLATFORM(DARWIN)#include <notify.h>#endif#if HAVE(SYS_TIME_H)#include <sys/time.h>#endif#if HAVE(SYS_TIMEB_H)#include <sys/timeb.h>#endif#if HAVE(STRINGS_H)#include <strings.h>#endifusing namespace WTF;namespace JSC {/* Constants */static const double minutesPerDay = 24.0 * 60.0;static const double secondsPerDay = 24.0 * 60.0 * 60.0;static const double secondsPerYear = 24.0 * 60.0 * 60.0 * 365.0;static const double usecPerSec = 1000000.0;static const double maxUnixTime = 2145859200.0; // 12/31/2037// Day of year for the first day of each month, where index 0 is January, and day 0 is January 1.// First for non-leap years, then for leap years.static const int firstDayOfMonth[2][12] = {    {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334},    {0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335}};static inline bool isLeapYear(int year){    if (year % 4 != 0)        return false;    if (year % 400 == 0)        return true;    if (year % 100 == 0)        return false;    return true;}static inline int daysInYear(int year){    return 365 + isLeapYear(year);}static inline double daysFrom1970ToYear(int year){    // The Gregorian Calendar rules for leap years:    // Every fourth year is a leap year.  2004, 2008, and 2012 are leap years.    // However, every hundredth year is not a leap year.  1900 and 2100 are not leap years.    // Every four hundred years, there's a leap year after all.  2000 and 2400 are leap years.    static const int leapDaysBefore1971By4Rule = 1970 / 4;    static const int excludedLeapDaysBefore1971By100Rule = 1970 / 100;    static const int leapDaysBefore1971By400Rule = 1970 / 400;    const double yearMinusOne = year - 1;    const double yearsToAddBy4Rule = floor(yearMinusOne / 4.0) - leapDaysBefore1971By4Rule;    const double yearsToExcludeBy100Rule = floor(yearMinusOne / 100.0) - excludedLeapDaysBefore1971By100Rule;    const double yearsToAddBy400Rule = floor(yearMinusOne / 400.0) - leapDaysBefore1971By400Rule;    return 365.0 * (year - 1970) + yearsToAddBy4Rule - yearsToExcludeBy100Rule + yearsToAddBy400Rule;}static inline double msToDays(double ms){    return floor(ms / msPerDay);}static inline int msToYear(double ms){    int approxYear = static_cast<int>(floor(ms / (msPerDay * 365.2425)) + 1970);    double msFromApproxYearTo1970 = msPerDay * daysFrom1970ToYear(approxYear);    if (msFromApproxYearTo1970 > ms)        return approxYear - 1;    if (msFromApproxYearTo1970 + msPerDay * daysInYear(approxYear) <= ms)        return approxYear + 1;    return approxYear;}static inline int dayInYear(double ms, int year){    return static_cast<int>(msToDays(ms) - daysFrom1970ToYear(year));}static inline double msToMilliseconds(double ms){    double result = fmod(ms, msPerDay);    if (result < 0)        result += msPerDay;    return result;}// 0: Sunday, 1: Monday, etc.static inline int msToWeekDay(double ms){    int wd = (static_cast<int>(msToDays(ms)) + 4) % 7;    if (wd < 0)        wd += 7;    return wd;}static inline int msToSeconds(double ms){    double result = fmod(floor(ms / msPerSecond), secondsPerMinute);    if (result < 0)        result += secondsPerMinute;    return static_cast<int>(result);}static inline int msToMinutes(double ms){    double result = fmod(floor(ms / msPerMinute), minutesPerHour);    if (result < 0)        result += minutesPerHour;    return static_cast<int>(result);}static inline int msToHours(double ms){    double result = fmod(floor(ms/msPerHour), hoursPerDay);    if (result < 0)        result += hoursPerDay;    return static_cast<int>(result);}static inline int monthFromDayInYear(int dayInYear, bool leapYear){    const int d = dayInYear;    int step;    if (d < (step = 31))        return 0;    step += (leapYear ? 29 : 28);    if (d < step)        return 1;    if (d < (step += 31))        return 2;    if (d < (step += 30))        return 3;    if (d < (step += 31))        return 4;    if (d < (step += 30))        return 5;    if (d < (step += 31))        return 6;    if (d < (step += 31))        return 7;    if (d < (step += 30))        return 8;    if (d < (step += 31))        return 9;    if (d < (step += 30))        return 10;    return 11;}static inline bool checkMonth(int dayInYear, int& startDayOfThisMonth, int& startDayOfNextMonth, int daysInThisMonth){    startDayOfThisMonth = startDayOfNextMonth;    startDayOfNextMonth += daysInThisMonth;    return (dayInYear <= startDayOfNextMonth);}static inline int dayInMonthFromDayInYear(int dayInYear, bool leapYear){    const int d = dayInYear;    int step;    int next = 30;    if (d <= next)        return d + 1;    const int daysInFeb = (leapYear ? 29 : 28);    if (checkMonth(d, step, next, daysInFeb))        return d - step;    if (checkMonth(d, step, next, 31))        return d - step;    if (checkMonth(d, step, next, 30))        return d - step;    if (checkMonth(d, step, next, 31))        return d - step;    if (checkMonth(d, step, next, 30))        return d - step;    if (checkMonth(d, step, next, 31))        return d - step;    if (checkMonth(d, step, next, 31))        return d - step;    if (checkMonth(d, step, next, 30))        return d - step;    if (checkMonth(d, step, next, 31))        return d - step;    if (checkMonth(d, step, next, 30))        return d - step;    step = next;    return d - step;}static inline int monthToDayInYear(int month, bool isLeapYear){    return firstDayOfMonth[isLeapYear][month];}static inline double timeToMS(double hour, double min, double sec, double ms){    return (((hour * minutesPerHour + min) * secondsPerMinute + sec) * msPerSecond + ms);}static int dateToDayInYear(int year, int month, int day){    year += month / 12;    month %= 12;    if (month < 0) {        month += 12;        --year;    }    int yearday = static_cast<int>(floor(daysFrom1970ToYear(year)));    int monthday = monthToDayInYear(month, isLeapYear(year));    return yearday + monthday + day - 1;}double getCurrentUTCTime(){    return floor(getCurrentUTCTimeWithMicroseconds());}// Returns current time in milliseconds since 1 Jan 1970.double getCurrentUTCTimeWithMicroseconds(){    return currentTime() * 1000.0; }void getLocalTime(const time_t* localTime, struct tm* localTM){#if COMPILER(MSVC7) || COMPILER(MINGW) || PLATFORM(WIN_CE)    *localTM = *localtime(localTime);#elif COMPILER(MSVC)    localtime_s(localTM, localTime);#else    localtime_r(localTime, localTM);#endif}// There is a hard limit at 2038 that we currently do not have a workaround// for (rdar://problem/5052975).static inline int maximumYearForDST(){    return 2037;}static inline int minimumYearForDST(){    // Because of the 2038 issue (see maximumYearForDST) if the current year is    // greater than the max year minus 27 (2010), we want to use the max year    // minus 27 instead, to ensure there is a range of 28 years that all years    // can map to.    return std::min(msToYear(getCurrentUTCTime()), maximumYearForDST() - 27) ;}/* * Find an equivalent year for the one given, where equivalence is deterined by * the two years having the same leapness and the first day of the year, falling * on the same day of the week. * * This function returns a year between this current year and 2037, however this * function will potentially return incorrect results if the current year is after * 2010, (rdar://problem/5052975), if the year passed in is before 1900 or after * 2100, (rdar://problem/5055038). */int equivalentYearForDST(int year){    // It is ok if the cached year is not the current year as long as the rules    // for DST did not change between the two years; if they did the app would need    // to be restarted.    static int minYear = minimumYearForDST();    int maxYear = maximumYearForDST();    int difference;    if (year > maxYear)        difference = minYear - year;    else if (year < minYear)        difference = maxYear - year;    else        return year;    int quotient = difference / 28;    int product = (quotient) * 28;    year += product;    ASSERT((year >= minYear && year <= maxYear) || (product - year == static_cast<int>(NaN)));    return year;}static int32_t calculateUTCOffset(){    tm localt;    memset(&localt, 0, sizeof(localt));     // get the difference between this time zone and UTC on Jan 01, 2000 12:00:00 AM    localt.tm_mday = 1;    localt.tm_year = 100;    time_t utcOffset = 946684800 - mktime(&localt);    return static_cast<int32_t>(utcOffset * 1000);}#if PLATFORM(DARWIN)static int32_t s_cachedUTCOffset; // In milliseconds. An assumption here is that access to an int32_t variable is atomic on platforms that take this code path.static bool s_haveCachedUTCOffset;static int s_notificationToken;#endif/* * Get the difference in milliseconds between this time zone and UTC (GMT) * NOT including DST. */double getUTCOffset(){#if PLATFORM(DARWIN)    if (s_haveCachedUTCOffset) {        int notified;        uint32_t status = notify_check(s_notificationToken, &notified);        if (status == NOTIFY_STATUS_OK && !notified)            return s_cachedUTCOffset;    }#endif    int32_t utcOffset = calculateUTCOffset();#if PLATFORM(DARWIN)    // Theoretically, it is possible that several threads will be executing this code at once, in which case we will have a race condition,    // and a newer value may be overwritten. In practice, time zones don't change that often.    s_cachedUTCOffset = utcOffset;#endif    return utcOffset;}/* * Get the DST offset for the time passed in.  Takes * seconds (not milliseconds) and cannot handle dates before 1970 * on some OS' */static double getDSTOffsetSimple(double localTimeSeconds, double utcOffset){    if (localTimeSeconds > maxUnixTime)        localTimeSeconds = maxUnixTime;    else if (localTimeSeconds < 0) // Go ahead a day to make localtime work (does not work with 0)        localTimeSeconds += secondsPerDay;    //input is UTC so we have to shift back to local time to determine DST thus the + getUTCOffset()    double offsetTime = (localTimeSeconds * msPerSecond) + utcOffset;    // Offset from UTC but doesn't include DST obviously    int offsetHour =  msToHours(offsetTime);    int offsetMinute =  msToMinutes(offsetTime);    // FIXME: time_t has a potential problem in 2038    time_t localTime = static_cast<time_t>(localTimeSeconds);    tm localTM;    getLocalTime(&localTime, &localTM);    double diff = ((localTM.tm_hour - offsetHour) * secondsPerHour) + ((localTM.tm_min - offsetMinute) * 60);    if (diff < 0)        diff += secondsPerDay;    return (diff * msPerSecond);}// Get the DST offset, given a time in UTCstatic double getDSTOffset(double ms, double utcOffset){    // On Mac OS X, the call to localtime (see getDSTOffsetSimple) will return historically accurate    // DST information (e.g. New Zealand did not have DST from 1946 to 1974) however the JavaScript    // standard explicitly dictates that historical information should not be considered when    // determining DST. For this reason we shift away from years that localtime can handle but would    // return historically accurate information.    int year = msToYear(ms);    int equivalentYear = equivalentYearForDST(year);    if (year != equivalentYear) {        bool leapYear = isLeapYear(year);        int dayInYearLocal = dayInYear(ms, year);        int dayInMonth = dayInMonthFromDayInYear(dayInYearLocal, leapYear);        int month = monthFromDayInYear(dayInYearLocal, leapYear);        int day = dateToDayInYear(equivalentYear, month, dayInMonth);        ms = (day * msPerDay) + msToMilliseconds(ms);    }    return getDSTOffsetSimple(ms / msPerSecond, utcOffset);}double gregorianDateTimeToMS(const GregorianDateTime& t, double milliSeconds, bool inputIsUTC){    int day = dateToDayInYear(t.year + 1900, t.month, t.monthDay);    double ms = timeToMS(t.hour, t.minute, t.second, milliSeconds);    double result = (day * msPerDay) + ms;    if (!inputIsUTC) { // convert to UTC        double utcOffset = getUTCOffset();        result -= utcOffset;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -