nginx的spinlock的学习

在nginx中,广泛使用了CAS来进行各个线程,各个进程的同步访问,而单独用CAS并不能最优的实现同步的功能,需要封装一个更合理的设计。

那么spinlock就设计出来了,采用了大部分用户态,偶尔内核态的方法,对频繁访问的数据进行了锁操作。

void
ngx_spinlock(ngx_atomic_t *lock, ngx_atomic_int_t value, ngx_uint_t spin)
{
    ngx_uint_t  i, n;

    for ( ;; ) {

        if (*lock == 0 && ngx_atomic_cmp_set(lock, 0, value)) {
            return;
        }

        if (ngx_ncpu > 1) {

            for (n = 1; n < spin; n <<= 1) {

                for (i = 0; i < n; i++) {
                    ngx_cpu_pause();
                }

                if (*lock == 0 && ngx_atomic_cmp_set(lock, 0, value)) {
                    return;
                }
            }
        }

        ngx_sched_yield();
    }
}
在spinlock中,第一次的ngx_atomic_cmp_set是单纯获取锁。
发现是多核cpu,ngx_ncpu > 1,则在用户态多等等。其中spin是设定的等待次数,ngx_cpu_pause是让cpu进行pause一下,即在用户态的可控的usleep。

如果是单核cpu,则直接ngx_sched_yield,让出cpu进入内核态,短暂的睡眠(sched_yield或者usleep)使线程后台运行,然后再在for(;;)中进行循环获取lock。这与mutex长期处于内核态sleep等待futex不同,spinlock的唤醒是主动的进入用户态,而不是像mutex那样被动的等待唤醒。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值