mirror of
https://gitlab.com/niansa/cosched.git
synced 2025-03-06 20:53:26 +01:00
Minor reworks and fixed lambda lifetime in ScheduledThread
This commit is contained in:
parent
78f710d0f5
commit
c016ccd672
5 changed files with 80 additions and 36 deletions
|
@ -14,7 +14,7 @@ namespace CoSched {
|
|||
class ScheduledThread {
|
||||
struct QueueEntry {
|
||||
std::string task_name;
|
||||
std::function<AwaitableTask<void> ()> task_fcn;
|
||||
std::function<AwaitableTask<void> ()> start_fcn;
|
||||
};
|
||||
|
||||
std::thread thread;
|
||||
|
@ -38,11 +38,11 @@ public:
|
|||
}
|
||||
|
||||
// DO NOT call from within a task
|
||||
void create_task(const std::string& task_name, const std::function<AwaitableTask<void> ()>& task_fcn) {
|
||||
void create_task(const std::string& task_name, std::function<AwaitableTask<void> ()>&& task_fcn) {
|
||||
// Enqueue function
|
||||
{
|
||||
std::scoped_lock L(queue_mutex);
|
||||
queue.emplace(QueueEntry{task_name, task_fcn});
|
||||
queue.emplace(QueueEntry{task_name, std::move(task_fcn)});
|
||||
}
|
||||
|
||||
// Notify thread
|
||||
|
|
|
@ -2,9 +2,11 @@
|
|||
#define _SCHEDULER_HPP
|
||||
#include <string>
|
||||
#include <vector>
|
||||
#include <unordered_map>
|
||||
#include <mutex>
|
||||
#include <memory>
|
||||
#include <chrono>
|
||||
#include <any>
|
||||
#include <AwaitableTask.hpp>
|
||||
#include <SingleEvent.hpp>
|
||||
|
||||
|
@ -24,10 +26,10 @@ enum {
|
|||
};
|
||||
|
||||
enum class TaskState {
|
||||
running,
|
||||
sleeping,
|
||||
terminating,
|
||||
dead
|
||||
running, // Task is currently in a normal running state
|
||||
sleeping, // Task is currently waiting to be scheduled again
|
||||
terminating, // Task will start terminating soon
|
||||
dead // Taks is currently terminating
|
||||
};
|
||||
|
||||
|
||||
|
@ -37,7 +39,7 @@ class Task {
|
|||
static thread_local class Task *current;
|
||||
|
||||
class Scheduler *scheduler;
|
||||
std::unique_ptr<SingleEvent<void>> resume_event;
|
||||
std::unique_ptr<SingleEvent<void>> resume_event = nullptr;
|
||||
|
||||
std::chrono::system_clock::time_point stopped_at;
|
||||
|
||||
|
@ -54,11 +56,16 @@ public:
|
|||
Task(const Task&) = delete;
|
||||
Task(Task&&) = delete;
|
||||
|
||||
// Misc property storage, unused
|
||||
std::unordered_map<std::string, std::any> properties;
|
||||
|
||||
// Returns the task that is currently being executed on this thread
|
||||
static inline
|
||||
Task& get_current() {
|
||||
return *current;
|
||||
}
|
||||
|
||||
// Sets a task name
|
||||
const std::string& get_name() const {
|
||||
return name;
|
||||
}
|
||||
|
@ -66,6 +73,7 @@ public:
|
|||
name = value;
|
||||
}
|
||||
|
||||
// Sets the task priority
|
||||
Priority get_priority() const {
|
||||
return priority;
|
||||
}
|
||||
|
@ -73,22 +81,22 @@ public:
|
|||
priority = value;
|
||||
}
|
||||
|
||||
// Returns the state of this task
|
||||
TaskState get_state() const {
|
||||
return state;
|
||||
}
|
||||
|
||||
// Returns the scheduler that is scheduling this task
|
||||
Scheduler& get_scheduler() const {
|
||||
return *scheduler;
|
||||
}
|
||||
|
||||
// Terminates the task as soon as possible
|
||||
void terminate() {
|
||||
if (state == TaskState::running) {
|
||||
state = TaskState::terminating;
|
||||
} else {
|
||||
state = TaskState::dead;
|
||||
}
|
||||
state = TaskState::terminating;
|
||||
}
|
||||
|
||||
// Suspends (pauses) the task as soon as possible
|
||||
void set_suspended(bool value = true) {
|
||||
suspended = value;
|
||||
}
|
||||
|
@ -96,6 +104,7 @@ public:
|
|||
return suspended;
|
||||
}
|
||||
|
||||
// Allows other tasks to execute
|
||||
AwaitableTask<bool> yield();
|
||||
};
|
||||
|
||||
|
|
|
@ -11,10 +11,15 @@ void CoSched::ScheduledThread::main_loop() {
|
|||
{
|
||||
std::scoped_lock L(queue_mutex);
|
||||
while (!queue.empty()) {
|
||||
// Get queue entry
|
||||
auto e = std::move(queue.front());
|
||||
queue.pop();
|
||||
// Create task for it
|
||||
sched.create_task(e.task_name);
|
||||
e.task_fcn();
|
||||
// Move start function somewhere else
|
||||
auto& start_fcn = std::any_cast<decltype(e.start_fcn)&>(Task::get_current().properties.emplace("start_function", std::move(e.start_fcn)).first->second);
|
||||
// Call start function
|
||||
start_fcn();
|
||||
}
|
||||
}
|
||||
// Run once
|
||||
|
|
|
@ -11,11 +11,16 @@ void CoSched::Task::kill() {
|
|||
}
|
||||
|
||||
AwaitableTask<bool> Task::yield() {
|
||||
// If it was terminating, it can finally be declared dead now
|
||||
if (state == TaskState::terminating) {
|
||||
// If it was terminating, it can finally be declared dead now
|
||||
state = TaskState::dead;
|
||||
co_return false;
|
||||
}
|
||||
// Dead tasks may not yield
|
||||
if (state == TaskState::dead) {
|
||||
co_return false;
|
||||
}
|
||||
if (this != current) co_return true;
|
||||
// It's just sleeping
|
||||
state = TaskState::sleeping;
|
||||
// Create event for resume
|
||||
|
@ -23,6 +28,13 @@ AwaitableTask<bool> Task::yield() {
|
|||
// Let's wait until we're back up!
|
||||
stopped_at = std::chrono::system_clock::now();
|
||||
co_await *resume_event;
|
||||
// Delete resume event
|
||||
resume_event = nullptr;
|
||||
// If task was terminating during sleep, it can finally be declared dead now
|
||||
if (state == TaskState::terminating) {
|
||||
state = TaskState::dead;
|
||||
co_return false;
|
||||
}
|
||||
// Here we go, let's keep going...
|
||||
state = TaskState::running;
|
||||
co_return true;
|
||||
|
@ -48,8 +60,8 @@ Task *Scheduler::get_next_task() {
|
|||
std::vector<Task*> max_prio_tasks;
|
||||
Priority max_prio = std::numeric_limits<Priority>::min();
|
||||
for (auto& task : tasks) {
|
||||
// Filter tasks that aren't sleeping
|
||||
if (task->state != TaskState::sleeping) continue;
|
||||
// Filter tasks can't currently be resumed
|
||||
if (task->resume_event == nullptr) continue;
|
||||
// Filter tasks that are suspended
|
||||
if (task->suspended) continue;
|
||||
// Update max priority
|
||||
|
|
56
test.cpp
56
test.cpp
|
@ -4,31 +4,49 @@
|
|||
#include <string>
|
||||
|
||||
|
||||
CoSched::AwaitableTask<std::string> get_value() {
|
||||
std::string fres = CoSched::Task::get_current().get_name();
|
||||
for (unsigned it = 0; it != 100; it++) {
|
||||
fres += "Hello";
|
||||
co_await CoSched::Task::get_current().yield();
|
||||
}
|
||||
fres.resize(1);
|
||||
co_return fres;
|
||||
}
|
||||
|
||||
CoSched::AwaitableTask<void> test_task() {
|
||||
auto& task = CoSched::Task::get_current();
|
||||
if (task.get_name() == "B" || task.get_name() == "D") {
|
||||
task.set_priority(CoSched::PRIO_HIGH);
|
||||
class LifetimeTest {
|
||||
std::string read_test_str = "Test value";
|
||||
|
||||
public:
|
||||
LifetimeTest() {
|
||||
std::cout << this << ": Lifetime start" << std::endl;
|
||||
}
|
||||
for (unsigned x = 100; co_await task.yield(); x--) {
|
||||
std::cout << co_await get_value() << ": " << x << '\n';
|
||||
if (x == 10) task.terminate();
|
||||
LifetimeTest(const LifetimeTest&) {
|
||||
std::cout << this << ": Lifetime copy" << std::endl;
|
||||
};
|
||||
LifetimeTest(LifetimeTest&&) {
|
||||
std::cout << this << ": Lifetime move" << std::endl;
|
||||
}
|
||||
}
|
||||
~LifetimeTest() {
|
||||
std::cout << this << ": Lifetime end" << std::endl;
|
||||
}
|
||||
|
||||
void read_test() const {
|
||||
std::cout << read_test_str << std::flush;
|
||||
std::cout << '\r';
|
||||
for (unsigned i = 0; i != read_test_str.size(); i++) {
|
||||
std::cout << ' ';
|
||||
}
|
||||
std::cout << '\r' << std::flush;
|
||||
std::cout << this << ": Lifetime read test success" << std::endl;
|
||||
}
|
||||
};
|
||||
|
||||
int main () {
|
||||
CoSched::ScheduledThread scheduler;
|
||||
for (const auto& name : {"A", "B", "C", "D", "E", "F"}) {
|
||||
scheduler.create_task(name, test_task);
|
||||
for (const auto& name : {"A", "B", "C"}) {
|
||||
scheduler.create_task(name, [lt = LifetimeTest()] () -> CoSched::AwaitableTask<void> {
|
||||
auto& task = CoSched::Task::get_current();
|
||||
std::cout << task.get_name() << "Scope start" << std::endl;
|
||||
lt.read_test();
|
||||
if (!co_await task.yield()) co_return;
|
||||
std::cout << task.get_name() << "Scope middle" << std::endl;
|
||||
lt.read_test();
|
||||
if (!co_await task.yield()) co_return;
|
||||
std::cout << task.get_name() << "Scope end" << std::endl;
|
||||
lt.read_test();
|
||||
});
|
||||
}
|
||||
scheduler.start();
|
||||
scheduler.wait();
|
||||
|
|
Loading…
Add table
Reference in a new issue