Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/kingluo/coco
coco is a simple stackless, single-threaded, and header-only C++11 coroutine library.
https://github.com/kingluo/coco
async-await channel coroutine cpp11 golang goroutine waitgroup
Last synced: 3 days ago
JSON representation
coco is a simple stackless, single-threaded, and header-only C++11 coroutine library.
- Host: GitHub
- URL: https://github.com/kingluo/coco
- Owner: kingluo
- License: mit
- Created: 2024-12-22T11:28:13.000Z (17 days ago)
- Default Branch: main
- Last Pushed: 2025-01-01T07:36:18.000Z (7 days ago)
- Last Synced: 2025-01-01T08:23:14.391Z (7 days ago)
- Topics: async-await, channel, coroutine, cpp11, golang, goroutine, waitgroup
- Language: C++
- Homepage: http://luajit.io/posts/coco-cpp11-coroutine/
- Size: 62.5 KB
- Stars: 10
- Watchers: 2
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# coco
coco is a simple stackless, single-threaded, and header-only C++11 coroutine library.
## Design
1. No depends on C++20 coroutine, C++11 is enough.
2. No compiler dependencies, simple macros, header-only.
3. coroutine like Lua, stackless async/await implementation just like Rust.
4. channel and waitgroup like Go.
5. single-threaded, no locks.
6. No performance overhead.## Synopsis
channel and waitgroup:
```cpp
auto sock_read = new co_t([=](co_t* __self, state_t* _st) {
auto st = dynamic_cast(_st);
COCO_ASYNC_BEGIN();
for (; st->cnt < 3; st->cnt++) {
COCO_ASYNC_BEGIN();
COCO_WRITE_CHAN(fs_write_ch, st->cnt);
COCO_WRITE_CHAN(kafka_produce_ch, st->cnt);
COCO_ASYNC_END();
}
fs_write_ch->close();
kafka_produce_ch->close();
COCO_WAIT(wg);
COCO_ASYNC_END();
COCO_DONE();
}, new sock_read_state_t);while (!sock_read->done())
sock_read->resume();```
webserver:
```cpp
go([=](co_t* __self, state_t* _st) {
auto st = dynamic_cast(_st);
st->co = __self;
while (true) {
COCO_ASYNC_BEGIN();
do_accept(server_socket, nullptr, nullptr, st);
COCO_YIELD();
auto sk = st->res;
go([=](co_t* __self, state_t* _st) {
auto req = dynamic_cast(_st);
req->co = __self;
COCO_ASYNC_BEGIN();read_request(req);
COCO_YIELD();handle_request(req);
send_response(req);
COCO_YIELD();
delete __self;COCO_ASYNC_END();
COCO_DONE();
}, new conn_state_t(sk));
COCO_ASYNC_END();
}
COCO_DONE();
}, new iouring_state_t);```
## Coding restrictions
Since Coco is simple and has no compiler support, it inevitably has some coding limitations.
1. coroutine lambda functions should be reentrant, so the statements should be reentrant and produce the same flow after reentry.
2. yield, wait, or channel r/w should be placed in an async block at the first level.
3. variables declared cannot escape yield, so if you need some variables to survive from a yield, use the state or global variables.
4. exceptions must be caught inside the lambda function.
5. state subclasses must inherit the `state_t` base class and `dynamic_cast` it to the subclass when using it.
6. The state is managed by the coroutine, so the state will be deleted when the coroutine exits.
7. You must delete all created coroutines, channels, and waitgroups yourself.