blob: 21910cffc376f75b20e438647d9dd6c9d1669549 [file] [log] [blame]
//===-- Unittests for mktime ----------------------------------------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "src/time/mktime.h"
#include "src/time/time_utils.h"
#include "test/UnitTest/ErrnoSetterMatcher.h"
#include "test/UnitTest/Test.h"
#include "test/src/time/TmHelper.h"
#include "test/src/time/TmMatcher.h"
#include <limits.h>
using __llvm_libc::testing::ErrnoSetterMatcher::Fails;
using __llvm_libc::testing::ErrnoSetterMatcher::Succeeds;
using __llvm_libc::time_utils::Month;
static inline constexpr int tm_year(int year) {
return year - TimeConstants::TIME_YEAR_BASE;
}
TEST(LlvmLibcMkTime, FailureSetsErrno) {
struct tm tm_data {
.tm_sec = INT_MAX, .tm_min = INT_MAX, .tm_hour = INT_MAX,
.tm_mday = INT_MAX, .tm_mon = INT_MAX - 1, .tm_year = tm_year(INT_MAX),
.tm_wday = 0, .tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data), Fails(EOVERFLOW));
}
TEST(LlvmLibcMkTime, InvalidSeconds) {
{
// -1 second from 1970-01-01 00:00:00 returns 1969-12-31 23:59:59.
struct tm tm_data {
.tm_sec = -1, .tm_min = 0, .tm_hour = 0, .tm_mday = 1,
.tm_mon = Month::JANUARY, .tm_year = tm_year(1970), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data), Succeeds(-1));
EXPECT_TM_EQ((tm{.tm_sec = 59,
.tm_min = 59,
.tm_hour = 23,
.tm_mday = 31,
.tm_mon = Month::DECEMBER,
.tm_year = tm_year(1969),
.tm_wday = 3,
.tm_yday = 364}),
tm_data);
}
{
// 60 seconds from 1970-01-01 00:00:00 returns 1970-01-01 00:01:00.
struct tm tm_data {
.tm_sec = 60, .tm_min = 0, .tm_hour = 0, .tm_mday = 1,
.tm_mon = Month::JANUARY, .tm_year = tm_year(1970), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data), Succeeds(60));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 1,
.tm_hour = 0,
.tm_mday = 1,
.tm_mon = Month::JANUARY,
.tm_year = tm_year(1970),
.tm_wday = 4,
.tm_yday = 0}),
tm_data);
}
}
TEST(LlvmLibcMkTime, InvalidMinutes) {
{
// -1 minute from 1970-01-01 00:00:00 returns 1969-12-31 23:59:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = -1, .tm_hour = 0, .tm_mday = 1,
.tm_mon = Month::JANUARY, .tm_year = tm_year(1970), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(-TimeConstants::SECONDS_PER_MIN));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 59,
.tm_hour = 23,
.tm_mday = 31,
.tm_mon = Month::DECEMBER,
.tm_year = tm_year(1969),
.tm_wday = 3,
.tm_yday = 0}),
tm_data);
}
{
// 60 minutes from 1970-01-01 00:00:00 returns 1970-01-01 01:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 60, .tm_hour = 0, .tm_mday = 1,
.tm_mon = Month::JANUARY, .tm_year = tm_year(1970), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(60 * TimeConstants::SECONDS_PER_MIN));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 1,
.tm_mday = 1,
.tm_mon = Month::JANUARY,
.tm_year = tm_year(1970),
.tm_wday = 4,
.tm_yday = 0}),
tm_data);
}
}
TEST(LlvmLibcMkTime, InvalidHours) {
{
// -1 hour from 1970-01-01 00:00:00 returns 1969-12-31 23:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 0, .tm_hour = -1, .tm_mday = 1,
.tm_mon = Month::JANUARY, .tm_year = tm_year(1970), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(-TimeConstants::SECONDS_PER_HOUR));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 23,
.tm_mday = 31,
.tm_mon = Month::DECEMBER,
.tm_year = tm_year(1969),
.tm_wday = 3,
.tm_yday = 0}),
tm_data);
}
{
// 24 hours from 1970-01-01 00:00:00 returns 1970-01-02 00:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 0, .tm_hour = 24, .tm_mday = 1,
.tm_mon = Month::JANUARY, .tm_year = tm_year(1970), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(24 * TimeConstants::SECONDS_PER_HOUR));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 0,
.tm_mday = 2,
.tm_mon = Month::JANUARY,
.tm_year = tm_year(1970),
.tm_wday = 5,
.tm_yday = 0}),
tm_data);
}
}
TEST(LlvmLibcMkTime, InvalidYear) {
// -1 year from 1970-01-01 00:00:00 returns 1969-01-01 00:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 0, .tm_hour = 0, .tm_mday = 1,
.tm_mon = Month::JANUARY, .tm_year = tm_year(1969), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(-TimeConstants::DAYS_PER_NON_LEAP_YEAR *
TimeConstants::SECONDS_PER_DAY));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 0,
.tm_mday = 1,
.tm_mon = Month::JANUARY,
.tm_year = tm_year(1969),
.tm_wday = 3,
.tm_yday = 0}),
tm_data);
}
TEST(LlvmLibcMkTime, InvalidEndOf32BitEpochYear) {
if (sizeof(size_t) != 4)
return;
{
// 2038-01-19 03:14:08 tests overflow of the second in 2038.
struct tm tm_data {
.tm_sec = 8, .tm_min = 14, .tm_hour = 3, .tm_mday = 19,
.tm_mon = Month::JANUARY, .tm_year = tm_year(2038), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(TimeConstants::OUT_OF_RANGE_RETURN_VALUE));
}
{
// 2038-01-19 03:15:07 tests overflow of the minute in 2038.
struct tm tm_data {
.tm_sec = 7, .tm_min = 15, .tm_hour = 3, .tm_mday = 19,
.tm_mon = Month::JANUARY, .tm_year = tm_year(2038), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(TimeConstants::OUT_OF_RANGE_RETURN_VALUE));
}
{
// 2038-01-19 04:14:07 tests overflow of the hour in 2038.
struct tm tm_data {
.tm_sec = 7, .tm_min = 14, .tm_hour = 4, .tm_mday = 19,
.tm_mon = Month::JANUARY, .tm_year = tm_year(2038), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(TimeConstants::OUT_OF_RANGE_RETURN_VALUE));
}
{
// 2038-01-20 03:14:07 tests overflow of the day in 2038.
struct tm tm_data {
.tm_sec = 7, .tm_min = 14, .tm_hour = 3, .tm_mday = 20,
.tm_mon = Month::JANUARY, .tm_year = tm_year(2038), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(TimeConstants::OUT_OF_RANGE_RETURN_VALUE));
}
{
// 2038-02-19 03:14:07 tests overflow of the month in 2038.
struct tm tm_data {
.tm_sec = 7, .tm_min = 14, .tm_hour = 3, .tm_mday = 19,
.tm_mon = Month::FEBRUARY, .tm_year = tm_year(2038), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(TimeConstants::OUT_OF_RANGE_RETURN_VALUE));
}
{
// 2039-01-19 03:14:07 tests overflow of the year.
struct tm tm_data {
.tm_sec = 7, .tm_min = 14, .tm_hour = 3, .tm_mday = 19,
.tm_mon = Month::JANUARY, .tm_year = tm_year(2039), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(TimeConstants::OUT_OF_RANGE_RETURN_VALUE));
}
}
TEST(LlvmLibcMkTime, InvalidMonths) {
{
// -1 month from 1970-01-01 00:00:00 returns 1969-12-01 00:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 0, .tm_hour = 0, .tm_mday = 0, .tm_mon = -1,
.tm_year = tm_year(1970), .tm_wday = 0, .tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(-32 * TimeConstants::SECONDS_PER_DAY));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 0,
.tm_mday = 1,
.tm_mon = Month::DECEMBER,
.tm_year = tm_year(1969),
.tm_wday = 1,
.tm_yday = 0}),
tm_data);
}
{
// 1970-13-01 00:00:00 returns 1971-01-01 00:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 0, .tm_hour = 0, .tm_mday = 1, .tm_mon = 12,
.tm_year = tm_year(1970), .tm_wday = 0, .tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(TimeConstants::DAYS_PER_NON_LEAP_YEAR *
TimeConstants::SECONDS_PER_DAY));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 0,
.tm_mday = 1,
.tm_mon = Month::JANUARY,
.tm_year = tm_year(1971),
.tm_wday = 5,
.tm_yday = 0}),
tm_data);
}
}
TEST(LlvmLibcMkTime, InvalidDays) {
{
// -1 day from 1970-01-01 00:00:00 returns 1969-12-31 00:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 0, .tm_hour = 0, .tm_mday = (1 - 1),
.tm_mon = Month::JANUARY, .tm_year = tm_year(1970), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(-1 * TimeConstants::SECONDS_PER_DAY));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 0,
.tm_mday = 31,
.tm_mon = Month::DECEMBER,
.tm_year = tm_year(1969),
.tm_wday = 3,
.tm_yday = 0}),
tm_data);
}
{
// 1970-01-32 00:00:00 returns 1970-02-01 00:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 0, .tm_hour = 0, .tm_mday = 32,
.tm_mon = Month::JANUARY, .tm_year = tm_year(1970), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(31 * TimeConstants::SECONDS_PER_DAY));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 0,
.tm_mday = 1,
.tm_mon = Month::FEBRUARY,
.tm_year = tm_year(1970),
.tm_wday = 0,
.tm_yday = 0}),
tm_data);
}
{
// 1970-02-29 00:00:00 returns 1970-03-01 00:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 0, .tm_hour = 0, .tm_mday = 29,
.tm_mon = Month::FEBRUARY, .tm_year = tm_year(1970), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(59 * TimeConstants::SECONDS_PER_DAY));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 0,
.tm_mday = 1,
.tm_mon = Month::MARCH,
.tm_year = tm_year(1970),
.tm_wday = 0,
.tm_yday = 0}),
tm_data);
}
{
// 1972-02-30 00:00:00 returns 1972-03-01 00:00:00.
struct tm tm_data {
.tm_sec = 0, .tm_min = 0, .tm_hour = 0, .tm_mday = 30,
.tm_mon = Month::FEBRUARY, .tm_year = tm_year(1972), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data),
Succeeds(((2 * TimeConstants::DAYS_PER_NON_LEAP_YEAR) + 60) *
TimeConstants::SECONDS_PER_DAY));
EXPECT_TM_EQ((tm{.tm_sec = 0,
.tm_min = 0,
.tm_hour = 0,
.tm_mday = 1,
.tm_mon = Month::MARCH,
.tm_year = tm_year(1972),
.tm_wday = 3,
.tm_yday = 0}),
tm_data);
}
}
TEST(LlvmLibcMkTime, EndOf32BitEpochYear) {
// Test for maximum value of a signed 32-bit integer.
// Test implementation can encode time for Tue 19 January 2038 03:14:07 UTC.
struct tm tm_data {
.tm_sec = 7, .tm_min = 14, .tm_hour = 3, .tm_mday = 19,
.tm_mon = Month::JANUARY, .tm_year = tm_year(2038), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data), Succeeds(0x7FFFFFFF));
EXPECT_TM_EQ((tm{.tm_sec = 7,
.tm_min = 14,
.tm_hour = 3,
.tm_mday = 19,
.tm_mon = Month::JANUARY,
.tm_year = tm_year(2038),
.tm_wday = 2,
.tm_yday = 7}),
tm_data);
}
TEST(LlvmLibcMkTime, Max64BitYear) {
if (sizeof(time_t) == 4)
return;
{
// Mon Jan 1 12:50:50 2170 (200 years from 1970),
struct tm tm_data {
.tm_sec = 50, .tm_min = 50, .tm_hour = 12, .tm_mday = 1,
.tm_mon = Month::JANUARY, .tm_year = tm_year(2170), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data), Succeeds(6311479850));
EXPECT_TM_EQ((tm{.tm_sec = 50,
.tm_min = 50,
.tm_hour = 12,
.tm_mday = 1,
.tm_mon = Month::JANUARY,
.tm_year = tm_year(2170),
.tm_wday = 1,
.tm_yday = 50}),
tm_data);
}
{
// Test for Tue Jan 1 12:50:50 in 2,147,483,647th year.
struct tm tm_data {
.tm_sec = 50, .tm_min = 50, .tm_hour = 12, .tm_mday = 1,
.tm_mon = Month::JANUARY, .tm_year = tm_year(2147483647), .tm_wday = 0,
.tm_yday = 0
};
EXPECT_THAT(__llvm_libc::mktime(&tm_data), Succeeds(67767976202043050));
EXPECT_TM_EQ((tm{.tm_sec = 50,
.tm_min = 50,
.tm_hour = 12,
.tm_mday = 1,
.tm_mon = Month::JANUARY,
.tm_year = tm_year(2147483647),
.tm_wday = 2,
.tm_yday = 50}),
tm_data);
}
}