xref: /openbmc/sdeventplus/test/source/time.cpp (revision 18db9a3e)
1a8c11e3cSPatrick Williams #include <systemd/sd-event.h>
2a8c11e3cSPatrick Williams #include <time.h>
3a8c11e3cSPatrick Williams 
4e32a88e1SWilliam A. Kennington III #include <sdeventplus/clock.hpp>
5e32a88e1SWilliam A. Kennington III #include <sdeventplus/exception.hpp>
6e32a88e1SWilliam A. Kennington III #include <sdeventplus/source/time.hpp>
7e32a88e1SWilliam A. Kennington III #include <sdeventplus/test/sdevent.hpp>
8a8c11e3cSPatrick Williams 
9a8c11e3cSPatrick Williams #include <cerrno>
10a8c11e3cSPatrick Williams #include <chrono>
11a8c11e3cSPatrick Williams #include <functional>
12a8c11e3cSPatrick Williams #include <memory>
13e32a88e1SWilliam A. Kennington III #include <utility>
14e32a88e1SWilliam A. Kennington III 
15a8c11e3cSPatrick Williams #include <gmock/gmock.h>
16a8c11e3cSPatrick Williams #include <gtest/gtest.h>
17a8c11e3cSPatrick Williams 
18e32a88e1SWilliam A. Kennington III namespace sdeventplus
19e32a88e1SWilliam A. Kennington III {
20e32a88e1SWilliam A. Kennington III namespace source
21e32a88e1SWilliam A. Kennington III {
22e32a88e1SWilliam A. Kennington III namespace
23e32a88e1SWilliam A. Kennington III {
24e32a88e1SWilliam A. Kennington III 
25e32a88e1SWilliam A. Kennington III using testing::DoAll;
26e32a88e1SWilliam A. Kennington III using testing::Return;
275320b1f5SWilliam A. Kennington III using testing::ReturnPointee;
28e32a88e1SWilliam A. Kennington III using testing::SaveArg;
29e32a88e1SWilliam A. Kennington III using testing::SetArgPointee;
30e32a88e1SWilliam A. Kennington III 
31e32a88e1SWilliam A. Kennington III using UniqueEvent = std::unique_ptr<Event, std::function<void(Event*)>>;
32e32a88e1SWilliam A. Kennington III 
33e32a88e1SWilliam A. Kennington III class TimeTest : public testing::Test
34e32a88e1SWilliam A. Kennington III {
35e32a88e1SWilliam A. Kennington III   protected:
36e32a88e1SWilliam A. Kennington III     testing::StrictMock<test::SdEventMock> mock;
37e32a88e1SWilliam A. Kennington III     sd_event_source* const expected_source =
38e32a88e1SWilliam A. Kennington III         reinterpret_cast<sd_event_source*>(1234);
39e32a88e1SWilliam A. Kennington III     sd_event* const expected_event = reinterpret_cast<sd_event*>(2345);
40e32a88e1SWilliam A. Kennington III     UniqueEvent event = make_event(expected_event);
41e32a88e1SWilliam A. Kennington III 
make_event(sd_event * event)42e32a88e1SWilliam A. Kennington III     UniqueEvent make_event(sd_event* event)
43e32a88e1SWilliam A. Kennington III     {
44e32a88e1SWilliam A. Kennington III         auto deleter = [this, event](Event* e) {
45e32a88e1SWilliam A. Kennington III             EXPECT_CALL(this->mock, sd_event_unref(event))
46e32a88e1SWilliam A. Kennington III                 .WillOnce(Return(nullptr));
47e32a88e1SWilliam A. Kennington III             delete e;
48e32a88e1SWilliam A. Kennington III         };
49e32a88e1SWilliam A. Kennington III         return UniqueEvent(new Event(event, std::false_type(), &mock), deleter);
50e32a88e1SWilliam A. Kennington III     }
51e32a88e1SWilliam A. Kennington III 
expect_time_destroy(sd_event * event,sd_event_source * source)52e32a88e1SWilliam A. Kennington III     void expect_time_destroy(sd_event* event, sd_event_source* source)
53e32a88e1SWilliam A. Kennington III     {
54e32a88e1SWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_source_unref(source))
55e32a88e1SWilliam A. Kennington III             .WillOnce(Return(nullptr));
56e32a88e1SWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_unref(event)).WillOnce(Return(nullptr));
57e32a88e1SWilliam A. Kennington III     }
58e32a88e1SWilliam A. Kennington III };
59e32a88e1SWilliam A. Kennington III 
TEST_F(TimeTest,ConstructSuccess)60e32a88e1SWilliam A. Kennington III TEST_F(TimeTest, ConstructSuccess)
61e32a88e1SWilliam A. Kennington III {
62e32a88e1SWilliam A. Kennington III     constexpr ClockId id = ClockId::RealTime;
63e32a88e1SWilliam A. Kennington III     const Time<id>::TimePoint expected_time(std::chrono::seconds{2});
64e32a88e1SWilliam A. Kennington III     const Time<id>::Accuracy expected_accuracy(std::chrono::milliseconds{50});
65e32a88e1SWilliam A. Kennington III     Time<id>::TimePoint saved_time;
66*18db9a3eSPatrick Williams     Time<id>::Callback callback =
67*18db9a3eSPatrick Williams         [&saved_time](Time<id>&, Time<id>::TimePoint time) {
68e32a88e1SWilliam A. Kennington III             saved_time = time;
69e32a88e1SWilliam A. Kennington III         };
70e32a88e1SWilliam A. Kennington III 
71e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock, sd_event_ref(expected_event))
72e32a88e1SWilliam A. Kennington III         .WillOnce(Return(expected_event));
73e32a88e1SWilliam A. Kennington III     sd_event_time_handler_t handler;
74e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock,
75e32a88e1SWilliam A. Kennington III                 sd_event_add_time(expected_event, testing::_, CLOCK_REALTIME,
76e32a88e1SWilliam A. Kennington III                                   2000000, 50000, testing::_, nullptr))
77e32a88e1SWilliam A. Kennington III         .WillOnce(DoAll(SetArgPointee<1>(expected_source), SaveArg<5>(&handler),
78e32a88e1SWilliam A. Kennington III                         Return(0)));
795320b1f5SWilliam A. Kennington III     sd_event_destroy_t destroy;
80e32a88e1SWilliam A. Kennington III     void* userdata;
815320b1f5SWilliam A. Kennington III     {
825320b1f5SWilliam A. Kennington III         testing::InSequence seq;
835320b1f5SWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_source_set_destroy_callback(expected_source,
845320b1f5SWilliam A. Kennington III                                                                testing::_))
855320b1f5SWilliam A. Kennington III             .WillOnce(DoAll(SaveArg<1>(&destroy), Return(0)));
865320b1f5SWilliam A. Kennington III         EXPECT_CALL(mock,
875320b1f5SWilliam A. Kennington III                     sd_event_source_set_userdata(expected_source, testing::_))
88e32a88e1SWilliam A. Kennington III             .WillOnce(DoAll(SaveArg<1>(&userdata), Return(nullptr)));
895320b1f5SWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_source_get_userdata(expected_source))
905320b1f5SWilliam A. Kennington III             .WillRepeatedly(ReturnPointee(&userdata));
915320b1f5SWilliam A. Kennington III     }
92e32a88e1SWilliam A. Kennington III     Time<id> time(*event, expected_time, expected_accuracy,
93e32a88e1SWilliam A. Kennington III                   std::move(callback));
941072c7f0SWilliam A. Kennington III     EXPECT_FALSE(callback);
955320b1f5SWilliam A. Kennington III     EXPECT_NE(&time, userdata);
96e32a88e1SWilliam A. Kennington III     EXPECT_EQ(expected_event, time.get_event().get());
97e32a88e1SWilliam A. Kennington III     EXPECT_EQ(expected_source, time.get());
98e32a88e1SWilliam A. Kennington III 
99e32a88e1SWilliam A. Kennington III     EXPECT_EQ(0, handler(nullptr, 2000100, userdata));
100e32a88e1SWilliam A. Kennington III     EXPECT_EQ(Time<id>::TimePoint(std::chrono::microseconds(2000100)),
101e32a88e1SWilliam A. Kennington III               saved_time);
102e32a88e1SWilliam A. Kennington III 
103b53ef902SWilliam A. Kennington III     time.set_callback(std::bind([]() {}));
104b53ef902SWilliam A. Kennington III     EXPECT_EQ(0, handler(nullptr, 0, userdata));
105b53ef902SWilliam A. Kennington III     EXPECT_EQ(Time<id>::TimePoint(std::chrono::microseconds(2000100)),
106b53ef902SWilliam A. Kennington III               saved_time);
107b53ef902SWilliam A. Kennington III 
108e32a88e1SWilliam A. Kennington III     expect_time_destroy(expected_event, expected_source);
1095320b1f5SWilliam A. Kennington III     destroy(userdata);
110e32a88e1SWilliam A. Kennington III }
111e32a88e1SWilliam A. Kennington III 
TEST_F(TimeTest,ConstructError)112e32a88e1SWilliam A. Kennington III TEST_F(TimeTest, ConstructError)
113e32a88e1SWilliam A. Kennington III {
114e32a88e1SWilliam A. Kennington III     constexpr ClockId id = ClockId::Monotonic;
115e32a88e1SWilliam A. Kennington III     const Time<id>::TimePoint expected_time(std::chrono::seconds{2});
116e32a88e1SWilliam A. Kennington III     const Time<id>::Accuracy expected_accuracy(std::chrono::milliseconds{50});
117e32a88e1SWilliam A. Kennington III     Time<id>::Callback callback = [](Time<id>&, Time<id>::TimePoint) {};
118e32a88e1SWilliam A. Kennington III 
119e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock,
120e32a88e1SWilliam A. Kennington III                 sd_event_add_time(expected_event, testing::_, CLOCK_MONOTONIC,
121e32a88e1SWilliam A. Kennington III                                   2000000, 50000, testing::_, nullptr))
122e32a88e1SWilliam A. Kennington III         .WillOnce(Return(-ENOSYS));
123*18db9a3eSPatrick Williams     EXPECT_THROW(Time<id>(*event, expected_time, expected_accuracy,
124*18db9a3eSPatrick Williams                           std::move(callback)),
125e32a88e1SWilliam A. Kennington III                  SdEventError);
1261072c7f0SWilliam A. Kennington III     EXPECT_TRUE(callback);
127e32a88e1SWilliam A. Kennington III }
128e32a88e1SWilliam A. Kennington III 
129e32a88e1SWilliam A. Kennington III class TimeMethodTest : public TimeTest
130e32a88e1SWilliam A. Kennington III {
131e32a88e1SWilliam A. Kennington III   protected:
132e32a88e1SWilliam A. Kennington III     static constexpr ClockId id = ClockId::BootTime;
133e32a88e1SWilliam A. Kennington III     std::unique_ptr<Time<id>> time;
1345320b1f5SWilliam A. Kennington III     sd_event_destroy_t destroy;
1355320b1f5SWilliam A. Kennington III     void* userdata;
136e32a88e1SWilliam A. Kennington III 
SetUp()137e32a88e1SWilliam A. Kennington III     void SetUp()
138e32a88e1SWilliam A. Kennington III     {
139e32a88e1SWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_ref(expected_event))
140e32a88e1SWilliam A. Kennington III             .WillOnce(Return(expected_event));
141e32a88e1SWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_add_time(expected_event, testing::_,
142e32a88e1SWilliam A. Kennington III                                             CLOCK_BOOTTIME, 2000000, 50000,
143e32a88e1SWilliam A. Kennington III                                             testing::_, nullptr))
144e32a88e1SWilliam A. Kennington III             .WillOnce(DoAll(SetArgPointee<1>(expected_source), Return(0)));
1455320b1f5SWilliam A. Kennington III         {
1465320b1f5SWilliam A. Kennington III             testing::InSequence seq;
1475320b1f5SWilliam A. Kennington III             EXPECT_CALL(mock, sd_event_source_set_destroy_callback(
1485320b1f5SWilliam A. Kennington III                                   expected_source, testing::_))
1495320b1f5SWilliam A. Kennington III                 .WillOnce(DoAll(SaveArg<1>(&destroy), Return(0)));
1505320b1f5SWilliam A. Kennington III             EXPECT_CALL(
1515320b1f5SWilliam A. Kennington III                 mock, sd_event_source_set_userdata(expected_source, testing::_))
1525320b1f5SWilliam A. Kennington III                 .WillOnce(DoAll(SaveArg<1>(&userdata), Return(nullptr)));
1535320b1f5SWilliam A. Kennington III             EXPECT_CALL(mock, sd_event_source_get_userdata(expected_source))
1545320b1f5SWilliam A. Kennington III                 .WillRepeatedly(ReturnPointee(&userdata));
1555320b1f5SWilliam A. Kennington III         }
156e32a88e1SWilliam A. Kennington III         time = std::make_unique<Time<id>>(
157e32a88e1SWilliam A. Kennington III             *event, Time<id>::TimePoint(std::chrono::seconds{2}),
158e32a88e1SWilliam A. Kennington III             std::chrono::milliseconds{50},
159e32a88e1SWilliam A. Kennington III             [](Time<id>&, Time<id>::TimePoint) {});
160e32a88e1SWilliam A. Kennington III     }
161e32a88e1SWilliam A. Kennington III 
TearDown()162e32a88e1SWilliam A. Kennington III     void TearDown()
163e32a88e1SWilliam A. Kennington III     {
164e32a88e1SWilliam A. Kennington III         expect_time_destroy(expected_event, expected_source);
165e32a88e1SWilliam A. Kennington III         time.reset();
1665320b1f5SWilliam A. Kennington III         destroy(userdata);
167e32a88e1SWilliam A. Kennington III     }
168e32a88e1SWilliam A. Kennington III };
169e32a88e1SWilliam A. Kennington III 
TEST_F(TimeMethodTest,Copy)17056dc78baSWilliam A. Kennington III TEST_F(TimeMethodTest, Copy)
17156dc78baSWilliam A. Kennington III {
17256dc78baSWilliam A. Kennington III     EXPECT_CALL(mock, sd_event_ref(expected_event))
17356dc78baSWilliam A. Kennington III         .WillOnce(Return(expected_event));
17456dc78baSWilliam A. Kennington III     EXPECT_CALL(mock, sd_event_source_ref(expected_source))
17556dc78baSWilliam A. Kennington III         .WillOnce(Return(expected_source));
17656dc78baSWilliam A. Kennington III     auto time2 = std::make_unique<Time<id>>(*time);
17756dc78baSWilliam A. Kennington III     {
17856dc78baSWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_ref(expected_event))
17956dc78baSWilliam A. Kennington III             .WillOnce(Return(expected_event));
18056dc78baSWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_source_ref(expected_source))
18156dc78baSWilliam A. Kennington III             .WillOnce(Return(expected_source));
18256dc78baSWilliam A. Kennington III         Time<id> time3(*time);
18356dc78baSWilliam A. Kennington III 
18456dc78baSWilliam A. Kennington III         expect_time_destroy(expected_event, expected_source);
18556dc78baSWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_ref(expected_event))
18656dc78baSWilliam A. Kennington III             .WillOnce(Return(expected_event));
18756dc78baSWilliam A. Kennington III         EXPECT_CALL(mock, sd_event_source_ref(expected_source))
18856dc78baSWilliam A. Kennington III             .WillOnce(Return(expected_source));
18956dc78baSWilliam A. Kennington III         *time2 = time3;
19056dc78baSWilliam A. Kennington III 
19156dc78baSWilliam A. Kennington III         expect_time_destroy(expected_event, expected_source);
19256dc78baSWilliam A. Kennington III     }
19356dc78baSWilliam A. Kennington III 
19456dc78baSWilliam A. Kennington III     // Delete the original time
19556dc78baSWilliam A. Kennington III     time2.swap(time);
19656dc78baSWilliam A. Kennington III     expect_time_destroy(expected_event, expected_source);
19756dc78baSWilliam A. Kennington III     time2.reset();
19856dc78baSWilliam A. Kennington III 
19956dc78baSWilliam A. Kennington III     // Make sure our new copy can still access data
20056dc78baSWilliam A. Kennington III     time->set_callback(nullptr);
20156dc78baSWilliam A. Kennington III }
20256dc78baSWilliam A. Kennington III 
TEST_F(TimeMethodTest,SetTimeSuccess)203e32a88e1SWilliam A. Kennington III TEST_F(TimeMethodTest, SetTimeSuccess)
204e32a88e1SWilliam A. Kennington III {
205e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock, sd_event_source_set_time(expected_source, 1000000))
206e32a88e1SWilliam A. Kennington III         .WillOnce(Return(0));
207e32a88e1SWilliam A. Kennington III     time->set_time(Time<id>::TimePoint(std::chrono::seconds{1}));
208e32a88e1SWilliam A. Kennington III }
209e32a88e1SWilliam A. Kennington III 
TEST_F(TimeMethodTest,SetTimeError)210e32a88e1SWilliam A. Kennington III TEST_F(TimeMethodTest, SetTimeError)
211e32a88e1SWilliam A. Kennington III {
212e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock, sd_event_source_set_time(expected_source, 1000000))
213e32a88e1SWilliam A. Kennington III         .WillOnce(Return(-EINVAL));
214e32a88e1SWilliam A. Kennington III     EXPECT_THROW(time->set_time(Time<id>::TimePoint(std::chrono::seconds{1})),
215e32a88e1SWilliam A. Kennington III                  SdEventError);
216e32a88e1SWilliam A. Kennington III }
217e32a88e1SWilliam A. Kennington III 
TEST_F(TimeMethodTest,GetTimeSuccess)218e32a88e1SWilliam A. Kennington III TEST_F(TimeMethodTest, GetTimeSuccess)
219e32a88e1SWilliam A. Kennington III {
220e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock, sd_event_source_get_time(expected_source, testing::_))
221e32a88e1SWilliam A. Kennington III         .WillOnce(DoAll(SetArgPointee<1>(10), Return(0)));
222e32a88e1SWilliam A. Kennington III     EXPECT_EQ(Time<id>::TimePoint(std::chrono::microseconds{10}),
223e32a88e1SWilliam A. Kennington III               time->get_time());
224e32a88e1SWilliam A. Kennington III }
225e32a88e1SWilliam A. Kennington III 
TEST_F(TimeMethodTest,GetTimeError)226e32a88e1SWilliam A. Kennington III TEST_F(TimeMethodTest, GetTimeError)
227e32a88e1SWilliam A. Kennington III {
228e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock, sd_event_source_get_time(expected_source, testing::_))
229e32a88e1SWilliam A. Kennington III         .WillOnce(Return(-ENOSYS));
230e32a88e1SWilliam A. Kennington III     EXPECT_THROW(time->get_time(), SdEventError);
231e32a88e1SWilliam A. Kennington III }
232e32a88e1SWilliam A. Kennington III 
TEST_F(TimeMethodTest,SetAccuracySuccess)233e32a88e1SWilliam A. Kennington III TEST_F(TimeMethodTest, SetAccuracySuccess)
234e32a88e1SWilliam A. Kennington III {
235e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock,
236e32a88e1SWilliam A. Kennington III                 sd_event_source_set_time_accuracy(expected_source, 5000000))
237e32a88e1SWilliam A. Kennington III         .WillOnce(Return(0));
238e32a88e1SWilliam A. Kennington III     time->set_accuracy(std::chrono::seconds{5});
239e32a88e1SWilliam A. Kennington III }
240e32a88e1SWilliam A. Kennington III 
TEST_F(TimeMethodTest,SetAccuracyError)241e32a88e1SWilliam A. Kennington III TEST_F(TimeMethodTest, SetAccuracyError)
242e32a88e1SWilliam A. Kennington III {
243e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock,
244e32a88e1SWilliam A. Kennington III                 sd_event_source_set_time_accuracy(expected_source, 5000000))
245e32a88e1SWilliam A. Kennington III         .WillOnce(Return(-EINVAL));
246e32a88e1SWilliam A. Kennington III     EXPECT_THROW(time->set_accuracy(std::chrono::seconds{5}), SdEventError);
247e32a88e1SWilliam A. Kennington III }
248e32a88e1SWilliam A. Kennington III 
TEST_F(TimeMethodTest,GetAccuracySuccess)249e32a88e1SWilliam A. Kennington III TEST_F(TimeMethodTest, GetAccuracySuccess)
250e32a88e1SWilliam A. Kennington III {
251e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock,
252e32a88e1SWilliam A. Kennington III                 sd_event_source_get_time_accuracy(expected_source, testing::_))
253e32a88e1SWilliam A. Kennington III         .WillOnce(DoAll(SetArgPointee<1>(1000), Return(0)));
254e32a88e1SWilliam A. Kennington III     EXPECT_EQ(std::chrono::milliseconds{1}, time->get_accuracy());
255e32a88e1SWilliam A. Kennington III }
256e32a88e1SWilliam A. Kennington III 
TEST_F(TimeMethodTest,GetAccuracyError)257e32a88e1SWilliam A. Kennington III TEST_F(TimeMethodTest, GetAccuracyError)
258e32a88e1SWilliam A. Kennington III {
259e32a88e1SWilliam A. Kennington III     EXPECT_CALL(mock,
260e32a88e1SWilliam A. Kennington III                 sd_event_source_get_time_accuracy(expected_source, testing::_))
261e32a88e1SWilliam A. Kennington III         .WillOnce(Return(-ENOSYS));
262e32a88e1SWilliam A. Kennington III     EXPECT_THROW(time->get_accuracy(), SdEventError);
263e32a88e1SWilliam A. Kennington III }
264e32a88e1SWilliam A. Kennington III 
265e32a88e1SWilliam A. Kennington III } // namespace
266e32a88e1SWilliam A. Kennington III } // namespace source
267e32a88e1SWilliam A. Kennington III } // namespace sdeventplus
268