xref: /aosp_15_r20/external/openscreen/platform/impl/time_unittest.cc (revision 3f982cf4871df8771c9d4abe6e9a6f8d829b2736)
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 Sanglard TEST(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