今天尝试着使用ucontext来实现一个轻量级的协程库,结果运行测试程序时,发生了段错误
通过gdb调试发现错误是Cannot access memory at address,不过这个信息对我的帮助太小
然后我加入了一些调试用的输出语句,就找到了真正的错误原因:我在创建一个协程时,需要传入一个执行函数,我对此函数进行了封装,作为makecontext的函数入口,这个封装函数会在原函数执行完毕后,做一些善后工作,譬如释放协程结构体占用的内存;可笑的是,我把协程使用的stack作为成员放入了这个结构体,那么当协程在运行时,这个stack指向的内存会受到保护,而我尝试去free它,那么就报了以上错误
struct Coroutine {
Scheduler *scher;
CoFunc func;
void *arg;
int state;
ucontext_t ctx;
char stack[DEFAULT_STACK_SZIE]; ----------------------> 协程运行使用的栈
};
// makecontext的函数入口
static void _coroutine_entry(uintptr_t *arg) {
Scheduler *s = (Scheduler *)arg;
assert(s != NULL);
int cid = s->running;
assert(cid >= 0 && cid < s->capacity);
Coroutine *co = s->cos[cid];
co->func(co->arg);
// coroutine done
free(co); ----------------------> 出错原因
s->cos[cid] = NULL;
s->running = -1;
}