1*3f982cf4SFabien Sanglard // Copyright 2019 The Chromium Authors. All rights reserved. 2*3f982cf4SFabien Sanglard // Use of this source code is governed by a BSD-style license that can be 3*3f982cf4SFabien Sanglard // found in the LICENSE file. 4*3f982cf4SFabien Sanglard 5*3f982cf4SFabien Sanglard #include "platform/api/time.h" 6*3f982cf4SFabien Sanglard 7*3f982cf4SFabien Sanglard #include <chrono> 8*3f982cf4SFabien Sanglard #include <ctime> 9*3f982cf4SFabien Sanglard 10*3f982cf4SFabien Sanglard #include "gtest/gtest.h" 11*3f982cf4SFabien Sanglard #include "util/chrono_helpers.h" 12*3f982cf4SFabien Sanglard 13*3f982cf4SFabien Sanglard namespace openscreen { 14*3f982cf4SFabien Sanglard 15*3f982cf4SFabien Sanglard #if __cplusplus < 202000L 16*3f982cf4SFabien Sanglard // Before C++20, the standard does not guarantee that time_t is the number of 17*3f982cf4SFabien Sanglard // seconds since the UNIX epoch. It also doesn't guarantee that one tick of 18*3f982cf4SFabien Sanglard // time_t represents one second. However, this is so universally common, it's 19*3f982cf4SFabien Sanglard // worth assuming the platform provides this. TEST(TimeTest,TimeTMeetsTheCpp20Standard)20*3f982cf4SFabien SanglardTEST(TimeTest, TimeTMeetsTheCpp20Standard) { 21*3f982cf4SFabien Sanglard // Use the functions the standard *does* provide for calendar time 22*3f982cf4SFabien Sanglard // conversions to determine the value of time_t at the UNIX epoch. 23*3f982cf4SFabien Sanglard std::tm epoch_tm; 24*3f982cf4SFabien Sanglard epoch_tm.tm_sec = 0; 25*3f982cf4SFabien Sanglard epoch_tm.tm_min = 0; 26*3f982cf4SFabien Sanglard epoch_tm.tm_hour = 0; 27*3f982cf4SFabien Sanglard epoch_tm.tm_mday = 1; 28*3f982cf4SFabien Sanglard epoch_tm.tm_mon = 0; 29*3f982cf4SFabien Sanglard epoch_tm.tm_year = 70; // 1970 30*3f982cf4SFabien Sanglard epoch_tm.tm_isdst = 0; 31*3f982cf4SFabien Sanglard // Note: std::mktime() assumes the translation is in the local time zone. 32*3f982cf4SFabien Sanglard const std::time_t new_year_1970_in_local_tz = std::mktime(&epoch_tm); 33*3f982cf4SFabien Sanglard // Purposely misinterpret |new_year_1970_in_local_tz| as UTC to provide 34*3f982cf4SFabien Sanglard // information as to how to offset |epoch_tm| such that std::mktime() will 35*3f982cf4SFabien Sanglard // be fooled into returning the value in terms of UTC. 36*3f982cf4SFabien Sanglard std::tm* const wrong_tm = std::gmtime(&new_year_1970_in_local_tz); 37*3f982cf4SFabien Sanglard epoch_tm.tm_sec += epoch_tm.tm_sec - wrong_tm->tm_sec; 38*3f982cf4SFabien Sanglard epoch_tm.tm_min += epoch_tm.tm_min - wrong_tm->tm_min; 39*3f982cf4SFabien Sanglard epoch_tm.tm_hour += epoch_tm.tm_hour - wrong_tm->tm_hour; 40*3f982cf4SFabien Sanglard epoch_tm.tm_mday += epoch_tm.tm_mday - wrong_tm->tm_mday; 41*3f982cf4SFabien Sanglard epoch_tm.tm_mon += epoch_tm.tm_mon - wrong_tm->tm_mon; 42*3f982cf4SFabien Sanglard epoch_tm.tm_year += epoch_tm.tm_year - wrong_tm->tm_year; 43*3f982cf4SFabien Sanglard 44*3f982cf4SFabien Sanglard const std::time_t epoch = std::mktime(&epoch_tm); 45*3f982cf4SFabien Sanglard EXPECT_EQ(seconds(0), seconds(epoch)); 46*3f982cf4SFabien Sanglard 47*3f982cf4SFabien Sanglard ++epoch_tm.tm_sec; 48*3f982cf4SFabien Sanglard const std::time_t epoch_plus_one_second = std::mktime(&epoch_tm); 49*3f982cf4SFabien Sanglard EXPECT_EQ(seconds(1), seconds(epoch_plus_one_second)); 50*3f982cf4SFabien Sanglard } 51*3f982cf4SFabien Sanglard #endif 52*3f982cf4SFabien Sanglard 53*3f982cf4SFabien Sanglard } // namespace openscreen 54