[libco] 删除协程的正确姿势

10 篇文章 0 订阅
删除libco协程时需谨慎,不能直接删除正在运行的协程,因协程可能在处理socket事件或定时器事件,此时删除可能导致野指针。正确的做法是在协程结束后(cEnd=1)安全删除。文章介绍了协程数据结构和运行函数,强调了删除工作中的协程的危险性,并展示了删除关注的socket事件和定时器事件的代码片段。
摘要由CSDN通过智能技术生成

如果你认为只需要简单调用 co_release 就能将 libco 的协程删除,那等待你的可能就是定时炸弹 💣。

文章来源:* [libco] 删除协程的正确姿势


1. 正确姿势

如何才能安全删除一个协程?

禁止删除一个正在工作的协程,删除已经停止工作(stCoRoutine_t.cEnd == 1)的协程是比较安全的。

/* 协程数据结构。 */
struct stCoRoutine_t {
    ...
    char cEnd; /* 协程是否结束。 */
    ...
};

/* 协程运行函数。 */
static int CoRoutineFunc(stCoRoutine_t *co, void *) {
    if (co->pfn) {
        co->pfn(co->arg);
    }
    co->cEnd = 1; /* 协程工作函数退出后,协程就已经结束了。 */

    stCoRoutineEnv_t *env = co->env;
    co_yield_env(env);
    return 0;
}

2. 原因

为啥删除工作中的协程是不安全?

因为协程在工作过程中可能触发 poll 功能。它主要处理了两种类型事件:socket 事件和定时器事件,这些事件都是异步回调的。当事件触发后,如果协程被释放了,那么保存的协程指针变成了 野指针!

int co_poll_inner(stCoEpoll_t *ctx, struct pollfd fds[], nfds_t nfds, int timeout, poll_pfn_t pollfunc) {
    ...
    stPoll_t &arg = *((stPoll_t *)malloc(sizeof(stPoll_t)));
    ...
    /* 保存当前协程指针。 */
    arg.pArg = GetCurrCo(co_get_curr_thread_env());
    ...
    /* 添加关注的 socket 事件。 */
    int ret = co_epoll_ctl(epfd, EPOLL_CTL_ADD, fds[i].fd, &ev);
    ...
    /* 添加定时器事件。 */
    int ret = AddTimeout(ctx->pTimeout, &arg, now);
    ...
    /* 切出当前协程。 */
    co_yield_env(co_get_curr_thread_env());
    ...
    /* 删除定时器事件。 */
    RemoveFromLink<stTimeoutItem_t, stTimeoutItemLink_t>(&arg);
    ...
    /* 删除关注的 socket 事件。 */
    co_epoll_ctl(epfd, EPOLL_CTL_DEL, fd, &arg.pPollItems[i].stEvent);
    ...
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值