blob: 0d6d5504d06a9e26ea6f3f1c1acfa4467022cfe9 [file] [log] [blame]
// Copyright 2023 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 "pw_async_basic/dispatcher.h"
#include <vector>
#include "pw_chrono/system_clock.h"
#include "pw_log/log.h"
#include "pw_span/span.h"
#include "pw_sync/thread_notification.h"
#include "pw_thread/thread.h"
#include "pw_thread_stl/options.h"
#include "pw_unit_test/framework.h"
#define ASSERT_CANCELLED(status) ASSERT_EQ(Status::Cancelled(), status)
using namespace std::chrono_literals;
namespace pw::async {
namespace {
// Lambdas can only capture one ptr worth of memory without allocating, so we
// group the data we want to share between tasks and their containing tests
// inside one struct.
struct TestPrimitives {
int count = 0;
sync::ThreadNotification notification;
};
TEST(DispatcherBasic, PostTasks) {
BasicDispatcher dispatcher;
Thread work_thread(thread::stl::Options(), dispatcher);
TestPrimitives tp;
auto inc_count = [&tp]([[maybe_unused]] Context& c, Status status) {
PW_TEST_ASSERT_OK(status);
++tp.count;
};
Task task(inc_count);
dispatcher.Post(task);
Task task2(inc_count);
dispatcher.Post(task2);
Task task3([&tp]([[maybe_unused]] Context& c, Status status) {
PW_TEST_ASSERT_OK(status);
++tp.count;
tp.notification.release();
});
dispatcher.Post(task3);
tp.notification.acquire();
dispatcher.RequestStop();
work_thread.join();
ASSERT_EQ(tp.count, 3);
}
TEST(DispatcherBasic, ChainedTasks) {
BasicDispatcher dispatcher;
Thread work_thread(thread::stl::Options(), dispatcher);
sync::ThreadNotification notification;
Task task1([&notification]([[maybe_unused]] Context& c, Status status) {
PW_TEST_ASSERT_OK(status);
notification.release();
});
Task task2([&task1](Context& c, Status status) {
PW_TEST_ASSERT_OK(status);
c.dispatcher->Post(task1);
});
Task task3([&task2](Context& c, Status status) {
PW_TEST_ASSERT_OK(status);
c.dispatcher->Post(task2);
});
dispatcher.Post(task3);
notification.acquire();
dispatcher.RequestStop();
work_thread.join();
}
TEST(DispatcherBasic, TaskOrdering) {
struct TestState {
std::vector<int> tasks;
sync::ThreadNotification notification;
};
BasicDispatcher dispatcher;
Thread work_thread(thread::stl::Options(), dispatcher);
TestState state;
Task task1([&state](Context&, Status status) {
PW_TEST_ASSERT_OK(status);
state.tasks.push_back(1);
});
Task task2([&state](Context&, Status status) {
PW_TEST_ASSERT_OK(status);
state.tasks.push_back(2);
state.notification.release();
});
// Task posted at same time should be ordered FIFO.
auto due_time = chrono::SystemClock::now();
dispatcher.PostAt(task1, due_time);
dispatcher.PostAt(task2, due_time);
state.notification.acquire();
dispatcher.RequestStop();
work_thread.join();
ASSERT_EQ(state.tasks.size(), 2U);
EXPECT_EQ(state.tasks[0], 1);
EXPECT_EQ(state.tasks[1], 2);
}
// Test RequestStop() from inside task.
TEST(DispatcherBasic, RequestStopInsideTask) {
BasicDispatcher dispatcher;
Thread work_thread(thread::stl::Options(), dispatcher);
int count = 0;
auto inc_count = [&count]([[maybe_unused]] Context& c, Status status) {
ASSERT_CANCELLED(status);
++count;
};
// These tasks are never executed and cleaned up in RequestStop().
Task task0(inc_count), task1(inc_count);
dispatcher.PostAfter(task0, 20ms);
dispatcher.PostAfter(task1, 21ms);
Task stop_task([&count]([[maybe_unused]] Context& c, Status status) {
PW_TEST_ASSERT_OK(status);
++count;
static_cast<BasicDispatcher*>(c.dispatcher)->RequestStop();
});
dispatcher.Post(stop_task);
work_thread.join();
ASSERT_EQ(count, 3);
}
TEST(DispatcherBasic, TasksCancelledByRequestStopInDifferentThread) {
BasicDispatcher dispatcher;
Thread work_thread(thread::stl::Options(), dispatcher);
int count = 0;
auto inc_count = [&count]([[maybe_unused]] Context& c, Status status) {
ASSERT_CANCELLED(status);
++count;
};
Task task0(inc_count), task1(inc_count), task2(inc_count);
dispatcher.PostAfter(task0, 10s);
dispatcher.PostAfter(task1, 10s);
dispatcher.PostAfter(task2, 10s);
dispatcher.RequestStop();
work_thread.join();
ASSERT_EQ(count, 3);
}
TEST(DispatcherBasic, TasksCancelledByDispatcherDestructor) {
int count = 0;
auto inc_count = [&count]([[maybe_unused]] Context& c, Status status) {
ASSERT_CANCELLED(status);
++count;
};
Task task0(inc_count), task1(inc_count), task2(inc_count);
{
BasicDispatcher dispatcher;
dispatcher.PostAfter(task0, 10s);
dispatcher.PostAfter(task1, 10s);
dispatcher.PostAfter(task2, 10s);
}
ASSERT_EQ(count, 3);
}
TEST(DispatcherBasic, TasksCancelledByRunUntilIdle) {
int count = 0;
auto inc_count = [&count]([[maybe_unused]] Context& c, Status status) {
ASSERT_CANCELLED(status);
++count;
};
Task task0(inc_count), task1(inc_count), task2(inc_count);
BasicDispatcher dispatcher;
dispatcher.PostAfter(task0, 10s);
dispatcher.PostAfter(task1, 10s);
dispatcher.PostAfter(task2, 10s);
dispatcher.RequestStop();
dispatcher.RunUntilIdle();
ASSERT_EQ(count, 3);
}
TEST(DispatcherBasic, TasksCancelledByRunFor) {
int count = 0;
auto inc_count = [&count]([[maybe_unused]] Context& c, Status status) {
ASSERT_CANCELLED(status);
++count;
};
Task task0(inc_count), task1(inc_count), task2(inc_count);
BasicDispatcher dispatcher;
dispatcher.PostAfter(task0, 10s);
dispatcher.PostAfter(task1, 10s);
dispatcher.PostAfter(task2, 10s);
dispatcher.RequestStop();
dispatcher.RunFor(5s);
ASSERT_EQ(count, 3);
}
class BasicDispatcherExecuteTask final : public BasicDispatcher {
public:
BasicDispatcherExecuteTask(pw::span<Task*> expected_tasks,
Status expected_status)
: expected_tasks_(expected_tasks), expected_status_(expected_status) {}
private:
void ExecuteTask(backend::NativeTask& task, Status status) final {
ASSERT_LT(executed_, expected_tasks_.size());
ASSERT_EQ(&task, &(expected_tasks_[executed_++]->native_type()));
ASSERT_EQ(status, expected_status_);
BasicDispatcher::ExecuteTask(task, status);
}
pw::span<Task*> expected_tasks_;
Status expected_status_;
uint32_t executed_ = 0;
};
TEST(DispatcherBasic, ExecuteTaskOk) {
TestPrimitives tp;
auto inc_count = [&tp]([[maybe_unused]] Context& c, Status status) {
PW_TEST_ASSERT_OK(status);
++tp.count;
};
Task task0(inc_count);
Task task1(inc_count);
Task task2([&tp]([[maybe_unused]] Context& c, Status status) {
PW_TEST_ASSERT_OK(status);
++tp.count;
tp.notification.release();
});
Task* tasks[] = {&task0, &task1, &task2};
BasicDispatcherExecuteTask dispatcher(tasks, OkStatus());
Thread work_thread(thread::stl::Options(), dispatcher);
dispatcher.Post(task0);
dispatcher.Post(task1);
dispatcher.Post(task2);
tp.notification.acquire();
dispatcher.RequestStop();
work_thread.join();
ASSERT_EQ(tp.count, 3);
}
TEST(DispatcherBasic, ExecuteTaskCancelled) {
int count = 0;
auto inc_count = [&count]([[maybe_unused]] Context& c, Status status) {
ASSERT_CANCELLED(status);
++count;
};
Task task0(inc_count);
Task task1(inc_count);
Task task2(inc_count);
Task* tasks[] = {&task0, &task1, &task2};
BasicDispatcherExecuteTask dispatcher(tasks, Status::Cancelled());
dispatcher.PostAfter(task0, 10s);
dispatcher.PostAfter(task1, 10s);
dispatcher.PostAfter(task2, 10s);
dispatcher.RequestStop();
dispatcher.RunFor(5s);
ASSERT_EQ(count, 3);
}
} // namespace
} // namespace pw::async