/ZERO

Coroutine for C++ with io_uring and work stealing scheduler as backend

Primary LanguageC++MIT LicenseMIT

ZERO

This project is aimed at creating a cpp coroutine framework including io_service based on io_uring and a work stealing scheduler for scheduling the coroutine.

Coroutine are function that can suspend execution and can be resumed later. This allow us to write programs where a thread dont have to wait for a result instead it can suspend the current coroutine and start executing another one and resume it once the result is available.

Coroutine

Coroutines in this framework use a threadpool managed by the scheduler. There can be more than one scheduler at a time running different coroutines. Coroutines are designed such that when a coroutine is resumed it resumes on the scheduler that it started on but it may not be on the same thread.

launch<T>

launch is used when we want to start our coroutine from a regular function. It is the entry point for our coroutines. launch require a scheduler to start running it is passed through schedule_on member function. When return type of launch is void the it will have a join member function to wait for its completion if not it will have a convertion operator which will cause the thread to wait till the result is available.

#include "coroutine/launch.hpp"
#include "coroutine/scheduler/scheduler.hpp"

launch<> hello() {
  std::cout << "Hello Coroutine\n";
  co_return;
}

launch<int> add(int a, int b) { co_return a + b; }

int main(int argc, char **argv) {
  scheduler schd;

  auto cr1 = hello().schedule_on(&schd); // Run coroutine on the scheduler
  cr1.join();                            // Wait for the coroutine to finish

  // When returning a value we can use convertion operator to wait and get the
  // value
  int cr2 = add(10, 20).schedule_on(&schd);
  std::cout << cr2 << std::endl;

  return 0;
}

task<T>

A task is one of the basic coroutine type. When a task is created it is suspended initially and return an awaiter to the caller. To run this task we have to co_await it. When the task is co_awaited the calling coroutine is suspended and the task is resumed and the calling coroutine is set as a continuation of the task, so when the task completes it can resume the suspended coroutine.

#include "coroutine/launch.hpp"
#include "coroutine/scheduler/scheduler.hpp"
#include "coroutine/task.hpp"
#include "io/io_service.hpp"

task<int> print_file(io_service &io, const std::string &filename) {
  char buffer[128]{0};
  int dir = open(".", 0);
  int fd  = co_await io.openat(dir, filename.c_str(), 0, 0);
  int len = co_await io.read(fd, buffer, 128, 0);
  int ret = co_await io.close(fd);
  std::cout << buffer << std::endl;
  co_return len;
}

launch<int> coroutine_1(io_service &io) {
  auto print_file_task = print_file(io, "log");
  co_return co_await print_file_task;
}

int main(int, char **) {
  scheduler scheduler;
  io_service io(100, 0);
  int len = coroutine_1(io).schedule_on(&scheduler);
  std::cout << "File Length : " << len << std::endl;
  return 0;
}

generator<T>

generator type allow us to yield a value and suspend the coroutine without exiting it. This allow us to resume to coroutine from where it is suspended.

#include "coroutine/generator.hpp"
#include "coroutine/launch.hpp"
#include "coroutine/scheduler/scheduler.hpp"

// Generate integers from zero
generator<int> integers() {
  int i   = 0;
  while (true) {
    co_yield i++;
  }
}

// Display an infinite number of integers
launch<int> launch_coroutine() {
  auto ints = integers();
  while (true) {
    std::cout << co_await ints << std::endl;
  }
  co_return 1;
}

int main(int argc, char **argv) {
  scheduler schd;
  int a = launch_coroutine().schedule_on(&schd);

  return 0;
}

async<T>

async coroutine start in suspended state. It can then be scheduled to run asynchronously by passing a scheduler to schedule_on member function or by co_await. When using co_await the scheduler used for current coroutine is used to schedule the async.

#include "coroutine/async.hpp"
#include "coroutine/launch.hpp"
#include "coroutine/scheduler/scheduler.hpp"
#include "io/io_service.hpp"

async<int> print_file(io_service &io, int dir, const std::string filename) {
  char buffer[256]{0};
  int fd = co_await io.openat(dir, filename.c_str(), 0, 0);
  co_return co_await io.read(fd, buffer, 256, 0);
}

launch<int> coroutine_1(io_service &io, int dir) {
  // Get the scheduler associated with this coroutine.
  auto scheduler       = co_await get_scheduler();
  auto print_file_task = print_file(io, dir, "log");
  co_return co_await print_file_task.schedule_on(scheduler);
}

launch<int> coroutine_2(io_service &io, int dir) {
  co_return co_await print_file(io, dir, "response");
}

int main(int, char **) {
  scheduler scheduler;
  io_service io(100, 0);

  int dir = open(".", 0);

  auto cr1 = coroutine_1(io, dir).schedule_on(&scheduler);
  auto cr2 = coroutine_2(io, dir).schedule_on(&scheduler);

  std::cout << "File Length 1 : " << cr1 << std::endl;
  std::cout << "File Length 2 : " << cr2 << std::endl;

  return 0;
}

timer

delay

delayed<T>

cancel

Coroutines support cooperative cancelation such that once a cancelation is requested the coroutine can decided to cancel it or not. It is implemented using stop_token so callbacks can be registered when the cancel request is recieved.

// This coroutine will suspend untill a connection is received and yield the connection fd.
generator<int> get_connections(io *io, int socket_fd) {
  // Get a stop_token for this coroutine.
  auto st = co_await get_stop_token();
  while (!st.stop_requested()) {  // Run untill a cancel is requested.
    sockaddr_in client_addr;
    socklen_t client_length;

    // Submit an io request for accepting a connection.
    auto awaiter =
        io->accept(socket_fd, (sockaddr *)&client_addr, &client_length, 0);

    // Register a stop_callback for this coroutine.
    std::stop_callback sc(st, [&] { io->cancel(awaiter, 0); });

    // Suspend this coroutine till a connection is received and yield the connection fd.
    co_yield co_await awaiter;
  }
  co_yield 0;
}

async<> server(io *io) {
  int socket_fd = get_tcp_socket();

  // Get a connection generator
  auto connections = get_connections(io, socket_fd);

  // Cancel the get_connection coroutine if this coroutine receives cancelation.
  std::stop_callback cb(co_await get_stop_token(),
                        [&] { connections.cancel(); });

  // Run untill get_connection coroutine is stoped.
  while (connections) {

    // Get a connection
    auto fd = co_await connections;
    if (fd <= 0) {
      co_return;
    }

    // Handle the client in another coroutine asynchronously
    handle_client(fd, io).schedule_on(co_await get_scheduler());
  }
  co_return;
}

schedule_on

event

Event allow us to wait for an event to happen. event provide a counter each time set is called this counter is incremented by the given value and when the event is co_awaited this value is returned and couter reset to zero.

async<void> print_task(event &e) {
  for (int i = 0; i < 10; ++i) {
    sleep(1);
    e.set(i);
  }
  co_return;
}

launch<int> launch_coroutine() {
  event e;
  print_task(e).schedule_on(co_await get_scheduler());
  for (int i = 0; i < 10; ++i) {
    std::cout << co_await e << std::endl;
  }
  co_return 0;
}

Scheduler

scheduler

A thread pool and a work stealing scheduler.

io

io_service

Wrapper for io_uring to support cpp coroutine. io_service have a dedicated thread for handling io and the io can be invoked from multiple thread.

Supported features

Chain Request

Batch Operation

Fixed Buffers

Provide Buffers

Supported io operations

openat

read

write

readv

writev

read_fixed

write_fixed

close

accept

send

recv

nop

timeout

link_timeout

cancel

statx

nop

delay

poll