//===-- Unittests for gmtime ----------------------------------------------===// // // 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/__support/CPP/limits.h" // INT_MAX, INT_MIN #include "src/errno/libc_errno.h" #include "src/time/gmtime.h" #include "src/time/time_utils.h" #include "test/UnitTest/ErrnoSetterMatcher.h" #include "test/UnitTest/Test.h" #include "test/src/time/TmMatcher.h" using LIBC_NAMESPACE::testing::ErrnoSetterMatcher::Fails; using LIBC_NAMESPACE::testing::ErrnoSetterMatcher::Succeeds; using LIBC_NAMESPACE::time_utils::TimeConstants; TEST(LlvmLibcGmTime, OutOfRange) { if (sizeof(time_t) < sizeof(int64_t)) return; time_t seconds = 1 + INT_MAX * static_cast( TimeConstants::NUMBER_OF_SECONDS_IN_LEAP_YEAR); struct tm *tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TRUE(tm_data == nullptr); ASSERT_ERRNO_EQ(EOVERFLOW); LIBC_NAMESPACE::libc_errno = 0; seconds = INT_MIN * static_cast( TimeConstants::NUMBER_OF_SECONDS_IN_LEAP_YEAR) - 1; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TRUE(tm_data == nullptr); ASSERT_ERRNO_EQ(EOVERFLOW); } TEST(LlvmLibcGmTime, InvalidSeconds) { time_t seconds = 0; struct tm *tm_data = nullptr; // -1 second from 1970-01-01 00:00:00 returns 1969-12-31 23:59:59. seconds = -1; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{59, // sec 59, // min 23, // hr 31, // day 12 - 1, // tm_mon starts with 0 for Jan 1969 - TimeConstants::TIME_YEAR_BASE, // year 3, // wday 364, // yday 0}), *tm_data); // 60 seconds from 1970-01-01 00:00:00 returns 1970-01-01 00:01:00. seconds = 60; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 1, // min 0, // hr 1, // day 0, // tm_mon starts with 0 for Jan 1970 - TimeConstants::TIME_YEAR_BASE, // year 4, // wday 0, // yday 0}), *tm_data); } TEST(LlvmLibcGmTime, InvalidMinutes) { time_t seconds = 0; struct tm *tm_data = nullptr; // -1 minute from 1970-01-01 00:00:00 returns 1969-12-31 23:59:00. seconds = -TimeConstants::SECONDS_PER_MIN; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 59, // min 23, // hr 31, // day 11, // tm_mon starts with 0 for Jan 1969 - TimeConstants::TIME_YEAR_BASE, // year 3, // wday 0, // yday 0}), *tm_data); // 60 minutes from 1970-01-01 00:00:00 returns 1970-01-01 01:00:00. seconds = 60 * TimeConstants::SECONDS_PER_MIN; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 1, // hr 1, // day 0, // tm_mon starts with 0 for Jan 1970 - TimeConstants::TIME_YEAR_BASE, // year 4, // wday 0, // yday 0}), *tm_data); } TEST(LlvmLibcGmTime, InvalidHours) { time_t seconds = 0; struct tm *tm_data = nullptr; // -1 hour from 1970-01-01 00:00:00 returns 1969-12-31 23:00:00. seconds = -TimeConstants::SECONDS_PER_HOUR; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 23, // hr 31, // day 11, // tm_mon starts with 0 for Jan 1969 - TimeConstants::TIME_YEAR_BASE, // year 3, // wday 0, // yday 0}), *tm_data); // 24 hours from 1970-01-01 00:00:00 returns 1970-01-02 00:00:00. seconds = 24 * TimeConstants::SECONDS_PER_HOUR; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 0, // hr 2, // day 0, // tm_mon starts with 0 for Jan 1970 - TimeConstants::TIME_YEAR_BASE, // year 5, // wday 0, // yday 0}), *tm_data); } TEST(LlvmLibcGmTime, InvalidYear) { // -1 year from 1970-01-01 00:00:00 returns 1969-01-01 00:00:00. time_t seconds = -TimeConstants::DAYS_PER_NON_LEAP_YEAR * TimeConstants::SECONDS_PER_DAY; struct tm *tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 0, // hr 1, // day 0, // tm_mon starts with 0 for Jan 1969 - TimeConstants::TIME_YEAR_BASE, // year 3, // wday 0, // yday 0}), *tm_data); } TEST(LlvmLibcGmTime, InvalidMonths) { time_t seconds = 0; struct tm *tm_data = nullptr; // -1 month from 1970-01-01 00:00:00 returns 1969-12-01 00:00:00. seconds = -31 * TimeConstants::SECONDS_PER_DAY; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 0, // hr 1, // day 12 - 1, // tm_mon starts with 0 for Jan 1969 - TimeConstants::TIME_YEAR_BASE, // year 1, // wday 0, // yday 0}), *tm_data); // 1970-13-01 00:00:00 returns 1971-01-01 00:00:00. seconds = TimeConstants::DAYS_PER_NON_LEAP_YEAR * TimeConstants::SECONDS_PER_DAY; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 0, // hr 1, // day 0, // tm_mon starts with 0 for Jan 1971 - TimeConstants::TIME_YEAR_BASE, // year 5, // wday 0, // yday 0}), *tm_data); } TEST(LlvmLibcGmTime, InvalidDays) { time_t seconds = 0; struct tm *tm_data = nullptr; // -1 day from 1970-01-01 00:00:00 returns 1969-12-31 00:00:00. seconds = -1 * TimeConstants::SECONDS_PER_DAY; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 0, // hr 31, // day 11, // tm_mon starts with 0 for Jan 1969 - TimeConstants::TIME_YEAR_BASE, // year 3, // wday 0, // yday 0}), *tm_data); // 1970-01-32 00:00:00 returns 1970-02-01 00:00:00. seconds = 31 * TimeConstants::SECONDS_PER_DAY; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 0, // hr 1, // day 0, // tm_mon starts with 0 for Jan 1970 - TimeConstants::TIME_YEAR_BASE, // year 0, // wday 0, // yday 0}), *tm_data); // 1970-02-29 00:00:00 returns 1970-03-01 00:00:00. seconds = 59 * TimeConstants::SECONDS_PER_DAY; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 0, // hr 1, // day 2, // tm_mon starts with 0 for Jan 1970 - TimeConstants::TIME_YEAR_BASE, // year 0, // wday 0, // yday 0}), *tm_data); // 1972-02-30 00:00:00 returns 1972-03-01 00:00:00. seconds = ((2 * TimeConstants::DAYS_PER_NON_LEAP_YEAR) + 60) * TimeConstants::SECONDS_PER_DAY; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{0, // sec 0, // min 0, // hr 1, // day 2, // tm_mon starts with 0 for Jan 1972 - TimeConstants::TIME_YEAR_BASE, // year 3, // wday 0, // yday 0}), *tm_data); } TEST(LlvmLibcGmTime, 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. time_t seconds = 0x7FFFFFFF; struct tm *tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{7, // sec 14, // min 3, // hr 19, // day 0, // tm_mon starts with 0 for Jan 2038 - TimeConstants::TIME_YEAR_BASE, // year 2, // wday 7, // yday 0}), *tm_data); } TEST(LlvmLibcGmTime, Max64BitYear) { if (sizeof(time_t) == 4) return; // Mon Jan 1 12:50:50 2170 (200 years from 1970), time_t seconds = 6311479850; struct tm *tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{50, // sec 50, // min 12, // hr 1, // day 0, // tm_mon starts with 0 for Jan 2170 - TimeConstants::TIME_YEAR_BASE, // year 1, // wday 50, // yday 0}), *tm_data); // Test for Tue Jan 1 12:50:50 in 2,147,483,647th year. seconds = 67767976202043050; tm_data = LIBC_NAMESPACE::gmtime(&seconds); EXPECT_TM_EQ((tm{50, // sec 50, // min 12, // hr 1, // day 0, // tm_mon starts with 0 for Jan 2147483647 - TimeConstants::TIME_YEAR_BASE, // year 2, // wday 50, // yday 0}), *tm_data); }