|  | // Copyright 2024 The Pigweed Authors | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); you may not | 
|  | // use this file except in compliance with the License. You may obtain a copy of | 
|  | // the License at | 
|  | // | 
|  | //     https://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | 
|  | // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | 
|  | // License for the specific language governing permissions and limitations under | 
|  | // the License. | 
|  |  | 
|  | #include "modules/blinky/blinky.h" | 
|  |  | 
|  | #include "pw_allocator/testing.h" | 
|  | #include "pw_async2/dispatcher.h" | 
|  | #include "pw_async2/simulated_time_provider.h" | 
|  | #include "pw_digital_io/digital_io_mock.h" | 
|  | #include "pw_unit_test/framework.h" | 
|  |  | 
|  | namespace demo { | 
|  |  | 
|  | using AllocatorForTest = ::pw::allocator::test::AllocatorForTest<512>; | 
|  | using ::pw::async2::Dispatcher; | 
|  | using ::pw::async2::SimulatedTimeProvider; | 
|  | using ::pw::chrono::SystemClock; | 
|  |  | 
|  | // Test fixtures. | 
|  |  | 
|  | class BlinkyTest : public ::testing::Test { | 
|  | protected: | 
|  | using Event = ::pw::digital_io::DigitalInOutMockImpl::Event; | 
|  | using State = ::pw::digital_io::State; | 
|  |  | 
|  | static constexpr uint32_t kIntervalMs = 10; | 
|  | static constexpr pw::chrono::SystemClock::duration kInterval = | 
|  | pw::chrono::SystemClock::for_at_least( | 
|  | std::chrono::milliseconds(kIntervalMs)); | 
|  |  | 
|  | BlinkyTest(): monochrome_led_(time_) { | 
|  | blinky_.Init(dispatcher_, time_, allocator_, monochrome_led_); | 
|  | } | 
|  |  | 
|  | pw::InlineDeque<Event>::iterator FirstActive() { | 
|  | pw::InlineDeque<Event>& events = monochrome_led_.events(); | 
|  | pw::InlineDeque<Event>::iterator event = events.begin(); | 
|  | while (event != events.end()) { | 
|  | if (event->state == State::kActive) { | 
|  | break; | 
|  | } | 
|  | ++event; | 
|  | } | 
|  | return event; | 
|  | } | 
|  |  | 
|  | uint32_t ToMs(pw::chrono::SystemClock::duration interval) { | 
|  | return std::chrono::duration_cast<std::chrono::milliseconds>(interval) | 
|  | .count(); | 
|  | } | 
|  |  | 
|  | static constexpr const size_t kEventCapacity = 256; | 
|  |  | 
|  | AllocatorForTest allocator_; | 
|  | Dispatcher dispatcher_; | 
|  | SimulatedTimeProvider<SystemClock> time_; | 
|  | pw::digital_io::DigitalInOutMock<kEventCapacity> monochrome_led_; | 
|  | Blinky blinky_; | 
|  | }; | 
|  |  | 
|  | // Unit tests. | 
|  |  | 
|  | TEST_F(BlinkyTest, Toggle) { | 
|  | auto start = time_.now(); | 
|  | blinky_.Toggle(); | 
|  | time_.AdvanceTime(kInterval * 1); | 
|  | blinky_.Toggle(); | 
|  | time_.AdvanceTime(kInterval * 2); | 
|  | blinky_.Toggle(); | 
|  | time_.AdvanceTime(kInterval * 3); | 
|  | blinky_.Toggle(); | 
|  |  | 
|  | auto event = FirstActive(); | 
|  | ASSERT_NE(event, monochrome_led_.events().end()); | 
|  | EXPECT_EQ(event->state, State::kActive); | 
|  | EXPECT_GE(ToMs(event->timestamp - start), kIntervalMs * 0); | 
|  | start = event->timestamp; | 
|  |  | 
|  | ASSERT_NE(++event, monochrome_led_.events().end()); | 
|  | EXPECT_EQ(event->state, State::kInactive); | 
|  | EXPECT_GE(ToMs(event->timestamp - start), kIntervalMs * 1); | 
|  | start = event->timestamp; | 
|  |  | 
|  | ASSERT_NE(++event, monochrome_led_.events().end()); | 
|  | EXPECT_EQ(event->state, State::kActive); | 
|  | EXPECT_GE(ToMs(event->timestamp - start), kIntervalMs * 2); | 
|  | start = event->timestamp; | 
|  |  | 
|  | ASSERT_NE(++event, monochrome_led_.events().end()); | 
|  | EXPECT_EQ(event->state, State::kInactive); | 
|  | EXPECT_GE(ToMs(event->timestamp - start), kIntervalMs * 3); | 
|  | } | 
|  |  | 
|  | TEST_F(BlinkyTest, Blink) { | 
|  | auto start = time_.now(); | 
|  | EXPECT_EQ(blinky_.Blink(1, kIntervalMs), pw::OkStatus()); | 
|  | while (!blinky_.IsIdle()) { | 
|  | dispatcher_.RunUntilStalled().IgnorePoll(); | 
|  | time_.AdvanceTime(kInterval); | 
|  | } | 
|  |  | 
|  | auto event = FirstActive(); | 
|  | ASSERT_NE(event, monochrome_led_.events().end()); | 
|  | EXPECT_EQ(event->state, State::kActive); | 
|  | EXPECT_GE(ToMs(event->timestamp - start), kIntervalMs); | 
|  | start = event->timestamp; | 
|  |  | 
|  | ASSERT_NE(++event, monochrome_led_.events().end()); | 
|  | EXPECT_EQ(event->state, State::kInactive); | 
|  | EXPECT_GE(ToMs(event->timestamp - start), kIntervalMs); | 
|  | } | 
|  |  | 
|  | TEST_F(BlinkyTest, BlinkMany) { | 
|  | auto start = time_.now(); | 
|  | EXPECT_EQ(blinky_.Blink(100, kIntervalMs), pw::OkStatus()); | 
|  | while (!blinky_.IsIdle()) { | 
|  | dispatcher_.RunUntilStalled().IgnorePoll(); | 
|  | time_.AdvanceTime(kInterval); | 
|  | } | 
|  |  | 
|  | // Every "on" and "off" is recorded. | 
|  | EXPECT_GE(monochrome_led_.events().size(), 200); | 
|  | EXPECT_GE(ToMs(time_.now() - start), kIntervalMs * 200); | 
|  | } | 
|  |  | 
|  | TEST_F(BlinkyTest, BlinkSlow) { | 
|  | auto start = time_.now(); | 
|  | EXPECT_EQ(blinky_.Blink(1, kIntervalMs * 32), pw::OkStatus()); | 
|  | while (!blinky_.IsIdle()) { | 
|  | dispatcher_.RunUntilStalled().IgnorePoll(); | 
|  | time_.AdvanceTime(kInterval); | 
|  | } | 
|  |  | 
|  | auto event = FirstActive(); | 
|  | ASSERT_NE(event, monochrome_led_.events().end()); | 
|  | EXPECT_EQ(event->state, State::kActive); | 
|  | EXPECT_GE(ToMs(event->timestamp - start), kIntervalMs * 32); | 
|  | start = event->timestamp; | 
|  |  | 
|  | ASSERT_NE(++event, monochrome_led_.events().end()); | 
|  | EXPECT_EQ(event->state, State::kInactive); | 
|  | EXPECT_GE(ToMs(event->timestamp - start), kIntervalMs * 32); | 
|  | } | 
|  |  | 
|  | }  // namespace demo |