Skip to content

Latest commit

 

History

History
147 lines (85 loc) · 4.79 KB

README.md

File metadata and controls

147 lines (85 loc) · 4.79 KB

Coroutine

中文文档:一个C++协程库的实现与优化

Abstract

A coroutine library implemented by C++, supporting both coroutine with shared runtime stack and coroutine with independent runtime stack.

Table of Contents

API

Schedule

Schedule::Schedule();

Schedule::Schedule(int size);

The first method creates a schedule with a 1MB runtime stack. All shared-stack coroutines created by the schedule will share this 1MB runtime stack.

The second method creates a schedule with a runtime stack whose size is determined by stack_size.

Even if all the coroutines managed by the schedule have independent runtime stacks, you must create a shared stack of at least 100 bytes in the schedule. In my design, all the coroutines will destroy their own stacks at the end of running, and the coroutines with independent runtime stack need to switch the runtime stack to the shared stack to destroy their own independent runtime stacks .

coroutine_create

int Schedule::coroutine_create(co_func func, void *args);

int Schedule::coroutine_create(co_func func, void *args, int stack_type, int stack_size);

The first method creates a coroutine with shared runtime stack, it equals to: Schedule::coroutine_create(func, args, SAVED_STACK, 0);

The second method creates a coroutine with shared runtime stack when stack_type is SAVED_STACK. stack_size determines the initial size of the memory saving coroutine's runtimestack, and the memory will be dynamically reallocated when the coroutine is suspended, so 0 is recomended for stack_size.

The second method creates a coroutine with independent runtime stack when stack_type is INDEPENDENT_STACK. stack_size determines the size(Bytes) of independent runtime stack.

coroutine_destroy

void Schedule::coroutine_destroy(int co_id);

This method will destroy the coroutine appointed by co_id.

coroutine_resume

void Schedule::coroutine_resume(int co_id);

This method switches the main coroutine to the non-main coroutine appointed by co_id.

coroutine_yield

void Schedule::coroutine_yield();

This method suspends the running non-main coroutine and switches to main coroutine.

coroutine_status

int Schedule::coroutine_status(int co_id);

This method returns the running coroutine status. The status(defined in coroutine.h) can be COROUTINE_DEAD, COROUTINE_READY, COROUTINE_RUNNING, COROUTINE_SUSPEND.

coroutine_running

int Schedule::coroutine_running() const;

This method returns the running coroutine id.

Testing Example

The test example is in test.cpp, and you can get more output of testing information by #define COROUTINE_TEST_OUTPUT.

Performance

This coroutine library provides two version of ucontext, The first one is defined in ucontext.h(glibc), and the second one defined in coctx.h(no FPU and MXCSR) the default choice, You can choose the first one by #define USE_SYS_UCONTEXT.

All non-main coroutine will return to main coroutine at the end of the run, coctx provides two mechanisms to return to the main coroutine. You can choose one of them by #define USE_UC_LINKor not.

Here are the performance testing data(switch 100,000,000 times):

ndef USE_SYS_UCONTEXT, def USE_UC_LINK, shared stack: 7.175s

ndef USE_SYS_UCONTEXT, ndef USE_UC_LINK, shared stack: 7.230s

def USE_SYS_UCONTEXT, shared stack: 91.81s

ndef USE_SYS_UCONTEXT, def USE_UC_LINK, independent stack: 5.749s

ndef USE_SYS_UCONTEXT, ndef USE_UC_LINK, independent stack: 5.697s

def USE_SYS_UCONTEXT independent stack: 89.52s

TODO

  • Try to add support to Block Syscall(maybe just read() and write()).

Suggestions

  • If the coroutines use shared runtime stack, don't use address on stack as buffer in async call! When one coroutine is suspended, the buffer allocated from runtime stack doesn't belong to it any more!

  • It's better to use Unblock Syscall than Block Syscall in coroutine.

Reference