mirror of
https://github.com/RGBCube/serenity
synced 2025-07-23 20:57:41 +00:00
AK+LibCore+Kernel: Have fewer implementations of day_of_year
The JS tests pointed out that the implementation in DateTime had an off-by-one in the month when doing the leap year check, so this change fixes that bug.
This commit is contained in:
parent
2c1b84b3e1
commit
c85e679e2d
6 changed files with 57 additions and 60 deletions
45
AK/Time.cpp
Normal file
45
AK/Time.cpp
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2020, The SerenityOS developers.
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||||
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <AK/Assertions.h>
|
||||||
|
#include <AK/Time.h>
|
||||||
|
|
||||||
|
namespace AK {
|
||||||
|
|
||||||
|
int day_of_year(int year, unsigned month, int day)
|
||||||
|
{
|
||||||
|
ASSERT(month >= 1 && month <= 12);
|
||||||
|
|
||||||
|
static const int seek_table[] = { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 };
|
||||||
|
int day_of_year = seek_table[month - 1] + day - 1;
|
||||||
|
|
||||||
|
if (is_leap_year(year) && month >= 3)
|
||||||
|
day_of_year++;
|
||||||
|
|
||||||
|
return day_of_year;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -28,6 +28,13 @@
|
||||||
|
|
||||||
namespace AK {
|
namespace AK {
|
||||||
|
|
||||||
|
// Month and day start at 1. Month must be >= 1 and <= 12.
|
||||||
|
// The return value is 0-indexed, that is Jan 1 is day 0.
|
||||||
|
// Day may be negative or larger than the number of days
|
||||||
|
// in the given month. If day is negative enough, the result
|
||||||
|
// can be negative.
|
||||||
|
int day_of_year(int year, unsigned month, int day);
|
||||||
|
|
||||||
inline bool is_leap_year(int year)
|
inline bool is_leap_year(int year)
|
||||||
{
|
{
|
||||||
return year % 4 == 0 && (year % 100 != 0 || year % 400 == 0);
|
return year % 4 == 0 && (year % 100 != 0 || year % 400 == 0);
|
||||||
|
@ -161,6 +168,7 @@ inline bool operator!=(const TimespecType& a, const TimespecType& b)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
using AK::day_of_year;
|
||||||
using AK::is_leap_year;
|
using AK::is_leap_year;
|
||||||
using AK::timespec_add;
|
using AK::timespec_add;
|
||||||
using AK::timespec_add_timeval;
|
using AK::timespec_add_timeval;
|
||||||
|
|
|
@ -207,6 +207,7 @@ set(AK_SOURCES
|
||||||
../AK/StringImpl.cpp
|
../AK/StringImpl.cpp
|
||||||
../AK/StringUtils.cpp
|
../AK/StringUtils.cpp
|
||||||
../AK/StringView.cpp
|
../AK/StringView.cpp
|
||||||
|
../AK/Time.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
set(ELF_SOURCES
|
set(ELF_SOURCES
|
||||||
|
|
|
@ -49,50 +49,6 @@ static bool update_in_progress()
|
||||||
return CMOS::read(0x0a) & 0x80;
|
return CMOS::read(0x0a) & 0x80;
|
||||||
}
|
}
|
||||||
|
|
||||||
static unsigned days_in_months_since_start_of_year(unsigned month, unsigned year)
|
|
||||||
{
|
|
||||||
ASSERT(month <= 11);
|
|
||||||
unsigned days = 0;
|
|
||||||
switch (month) {
|
|
||||||
case 11:
|
|
||||||
days += 30;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 10:
|
|
||||||
days += 31;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 9:
|
|
||||||
days += 30;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 8:
|
|
||||||
days += 31;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 7:
|
|
||||||
days += 31;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 6:
|
|
||||||
days += 30;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 5:
|
|
||||||
days += 31;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 4:
|
|
||||||
days += 30;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 3:
|
|
||||||
days += 31;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 2:
|
|
||||||
if (is_leap_year(year))
|
|
||||||
days += 29;
|
|
||||||
else
|
|
||||||
days += 28;
|
|
||||||
[[fallthrough]];
|
|
||||||
case 1:
|
|
||||||
days += 31;
|
|
||||||
}
|
|
||||||
return days;
|
|
||||||
}
|
|
||||||
|
|
||||||
static u8 bcd_to_binary(u8 bcd)
|
static u8 bcd_to_binary(u8 bcd)
|
||||||
{
|
{
|
||||||
return (bcd & 0x0F) + ((bcd >> 4) * 10);
|
return (bcd & 0x0F) + ((bcd >> 4) * 10);
|
||||||
|
@ -149,8 +105,7 @@ time_t now()
|
||||||
ASSERT(year >= 2018);
|
ASSERT(year >= 2018);
|
||||||
|
|
||||||
return years_to_days_since_epoch(year) * 86400
|
return years_to_days_since_epoch(year) * 86400
|
||||||
+ days_in_months_since_start_of_year(month - 1, year) * 86400
|
+ day_of_year(year, month, day) * 86400
|
||||||
+ (day - 1) * 86400
|
|
||||||
+ hour * 3600
|
+ hour * 3600
|
||||||
+ minute * 60
|
+ minute * 60
|
||||||
+ second;
|
+ second;
|
||||||
|
|
|
@ -117,13 +117,7 @@ static time_t tm_to_time(struct tm* tm, long timezone_adjust_seconds)
|
||||||
}
|
}
|
||||||
|
|
||||||
int days = years_to_days_since_epoch(1900 + tm->tm_year);
|
int days = years_to_days_since_epoch(1900 + tm->tm_year);
|
||||||
|
tm->tm_yday = day_of_year(1900 + tm->tm_year, tm->tm_mon + 1, tm->tm_mday);
|
||||||
tm->tm_yday = tm->tm_mday - 1;
|
|
||||||
for (int month = 0; month < tm->tm_mon; ++month)
|
|
||||||
tm->tm_yday += __days_per_month[month];
|
|
||||||
if (tm->tm_mon > 1 && is_leap_year(1900 + tm->tm_year))
|
|
||||||
++tm->tm_yday;
|
|
||||||
|
|
||||||
days += tm->tm_yday;
|
days += tm->tm_yday;
|
||||||
|
|
||||||
int seconds = tm->tm_hour * 3600 + tm->tm_min * 60 + tm->tm_sec;
|
int seconds = tm->tm_hour * 3600 + tm->tm_min * 60 + tm->tm_sec;
|
||||||
|
|
|
@ -81,13 +81,7 @@ unsigned DateTime::days_in_month() const
|
||||||
|
|
||||||
unsigned DateTime::day_of_year() const
|
unsigned DateTime::day_of_year() const
|
||||||
{
|
{
|
||||||
static const int seek_table[] = { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 };
|
return ::day_of_year(m_year, m_month, m_day);
|
||||||
int day_of_year = seek_table[m_month - 1] + m_day;
|
|
||||||
|
|
||||||
if (is_leap_year() && m_month > 3)
|
|
||||||
day_of_year++;
|
|
||||||
|
|
||||||
return day_of_year - 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DateTime::is_leap_year() const
|
bool DateTime::is_leap_year() const
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue