Linux 中的同步机制

news2024/9/23 17:21:36

代码基于:Kernel 6.6

临界资源:指哪些在同一时刻只允许被一个线程访问的软件或硬件资源。这种资源的特点是,如果有线程正在使用,其他进程必须等待直到该线程释放资源。

临界区:指在每个线程中访问临界资源的那段代码。由于临界资源需要互斥访问,因此每个线程的临界区代码在执行时需要确保没有其他现成同事访问该资源。

1. 原子操作(atomic)

原子操作是指不能再进一步分割的操作,是指单个处理器执行期间,一组操作被视为不可分割的整理,要么全部完成,要么全部不执行。这些操作不会被其他线程或中断打断,因此保证数据的一致性和完整性。

原子操作特点:

  • 封装性:操作内部的细节外部不可见,保证了操作的可见性和顺序一致性;
  • 隔离性:在同一时间点内,只有单个线程能够执行原子操作,避免了竞态条件;
  • 原子性:在并发环境中,整个操作表现为不可分割的单位;
include/linux/types.h

typedef struct {
        int counter;
} atomic_t;

#define ATOMIC_INIT(i) { (i) }    //atomic_t 变量初始化

#ifdef CONFIG_64BIT
typedef struct {
        s64 counter;
} atomic64_t;
#endif
include/asm-generic/atomic.h

#define arch_atomic_read(v)                        READ_ONCE((v)->counter)
#define arch_atomic_set(v, i)                      WRITE_ONCE(((v)->counter), (i))

一般会使用atomic_readatomic_set接口对atomic 变量进行读写,不同使用场景这两个函数的实现也不同,例如使能 CONFIG_ARCH_ATOMIC 会调用上面的 arch_atomic_read

在 SMP 系统中,atomic_read接口调用的就是READ_ONCE()atomic_set接口调用的就是WRITE_ONCE()函数。

2. 互斥锁(Mutex)

互斥锁是一种休眠锁,锁征用时可能存在进程的睡眠与唤醒,context 的切换带来的代价高,适用于加锁时间较长的场景。

互斥锁每次只允许一个进程进入临界区,有点类似二元信号量。也只能是互斥锁的 owner可以unlock。

与信号量相比,互斥锁的性能与扩展性都更好,因此在内核中总是会优先考虑互斥锁。

include/linux/mutex.h

struct mutex {
        atomic_long_t               owner;        //原子计数,用于指向锁持有者的 task_struct
        raw_spinlock_t              wait_lock;    //自旋锁,用于 wait_list 链表的互斥
#ifdef CONFIG_MUTEX_SPIN_ON_OWNER
        struct optimistic_spin_queue osq;         //osq 锁
#endif
        struct list_head        wait_list;        //链表,用于管理所有在该mutex 上睡眠的线程
#ifdef CONFIG_DEBUG_MUTEXES
        void                        *magic;
#endif
#ifdef CONFIG_DEBUG_LOCK_ALLOC
        struct lockdep_map          dep_map;
#endif
};
kernel/locking/mutex.h

struct mutex_waiter {
        struct list_head        list;     //用以添加到wait_list
        struct task_struct      *task;    //等待的线程
        struct ww_acquire_ctx   *ww_ctx;
#ifdef CONFIG_DEBUG_MUTEXES
        void                    *magic;
#endif
};

在使用mutex 时,有如下注意点:

  • 一次只能有一个进程持有互斥锁;
  • 只有锁的 owner能进行unlock 操作;
  • 禁止多次解锁操作;
  • 禁止递归加锁操作;
  • mutex 数据结构只能通过API 进行初始化;
  • mutex 数据结构禁止使用memset或者copy来进行初始化;
  • 已经被持有的mutex锁,禁止被再次初始化;
  • 持有mutex 锁的task 不能exit
  • 持有mutex 锁的内存区域不能被free
  • mutex 锁不允许在硬件、软件中断上下文,例如tasklets、timers 等;

2.1 mutext 初始化

include/linux/mutex.h

#define mutex_init(mutex)                                                \
do {                                                                     \
        static struct lock_class_key __key;                              \
                                                                         \
        __mutex_init((mutex), #mutex, &__key);                           \
} while (0)
kernel/locking/mutex.c

void
__mutex_init(struct mutex *lock, const char *name, struct lock_class_key *key)
{
        atomic_long_set(&lock->owner, 0);
        raw_spin_lock_init(&lock->wait_lock);
        INIT_LIST_HEAD(&lock->wait_list);
#ifdef CONFIG_MUTEX_SPIN_ON_OWNER
        osq_lock_init(&lock->osq);
#endif

        debug_mutex_init(lock, name, key);
}
EXPORT_SYMBOL(__mutex_init);

调用初始化的时候,会将owner 重置,wai_lock、wait_list、osq 等都会被重置。这就是为什么强调被初始化的mutex 不能再初始化(schedule 的线程可能再也无法唤醒)。

2.2 mutex_lock()

include/linux/mutex.h

extern void mutex_lock(struct mutex *lock);

该图是mutex_lock函数调用的大致流程图,从流程图中可以看出该函数大致分两步:

  • 快速lock
  • 慢速lock

__mutex_trylock_fast函数会调用 atomic_long_try_cmpxchg_acquire函数,true 代表锁成功。

atomic_long_try_cmpxchg_acquire函数中判断lock->owner==0表明锁未持有,将curr赋值给lock->owner标记curr 进程持有该锁,并返回true;若lock->owner !=0 表明锁被持有,则返回false 继续慢速 lock。

进入慢速路径最终调用的是__mutex_lock_common函数,该过程比较复杂,感兴趣可以查看源码。

2.3 mutex_unlock()

include/linux/mutex.h

extern void mutex_unlock(struct mutex *lock);

释放锁的过程比较简单,核心调用__mutex_unlock_slowpath函数,当 wait_list 不为空时会取出第一个mutex_waiter,并调用wake_q_add将该waiter 的task 添加到唤醒队列,最终调用wake_up_q函数唤醒等在该锁上的线程。

参考:

https://zhuanlan.zhihu.com/p/633069533

https://www.cnblogs.com/LoyenWang/p/12826811.html

3. 信号量(semaphore)

信号量是Linux 系统常用的同步与互斥的机制。

信号量在创建时需要设置一个初始值,表示同时可以有几个任务访问该信号量保护的共享资源。当初始值为1,表示同时只能有一个任务可以访问信号量保护的共享资源,该信号量又称为二元信号量(bindary semaphore).

可以将信号量比喻成一个盒子,初始化时在盒子里放入N把钥匙,钥匙先到先得,当N把钥匙都被拿走完后,再来拿钥匙的人就需要等待了,只有等到有人将钥匙归还了,等待的人才能拿到钥匙。

inlucde/linux/semaphore.h

struct semaphore {
        raw_spinlock_t          lock;       //自旋锁,用于count值的互斥访问
        unsigned int            count;      //计数值,能同时允许访问的数量,也就是上文中的 N把钥匙
        struct list_head        wait_list;  //不能立即获取到信号量的访问者,都会加到等待队列中
};
kernel/locking/semaphore.c

struct semaphore_waiter {  //当 N 把钥匙被拿走,后来的进程需要等待
        struct list_head   list;     //用于添加到 wait_list中
        struct task_struct *task;    //等待的线程
        bool up;                     //标记是否释放信号量
};

3.1 信号量的初始化

inlucde/linux/semaphore.h

static inline void sema_init(struct semaphore *sem, int val)
{
        static struct lock_class_key __key;
        *sem = (struct semaphore) __SEMAPHORE_INITIALIZER(*sem, val);
        lockdep_init_map(&sem->lock.dep_map, "semaphore->lock", &__key, 0);
}

#define DEFINE_SEMAPHORE(_name, _n)        \
        struct semaphore _name = __SEMAPHORE_INITIALIZER(_name, _n)

信号量的初始化有两种方式,sema_init 接口需要在 CONFIG_DEBUG_LOCK_ALLOC 使能的情况下使用。

3.2 down/up

inlucde/linux/semaphore.h

extern void down(struct semaphore *sem);
extern void up(struct semaphore *sem);
kernel/locking/semaphore.c

void __sched down(struct semaphore *sem)
{
        unsigned long flags;

        might_sleep();
        raw_spin_lock_irqsave(&sem->lock, flags);
        if (likely(sem->count > 0))
                sem->count--;
        else
                __down(sem);
        raw_spin_unlock_irqrestore(&sem->lock, flags);
}
EXPORT_SYMBOL(down);

void __sched up(struct semaphore *sem)
{
        unsigned long flags;

        raw_spin_lock_irqsave(&sem->lock, flags);
        if (likely(list_empty(&sem->wait_list)))
                sem->count++;
        else
                __up(sem);
        raw_spin_unlock_irqrestore(&sem->lock, flags);
}
EXPORT_SYMBOL(up);

  • down接口用于获取信号量,up用于释放信号量;
  • 调用down时,如果sem->count > 0时,也就是盒子里边还有多余的锁,直接自减并返回了,当sem->count == 0时,表明盒子里边的锁被用完了,当前任务会加入信号量的等待列表中,设置进程的状态,并调用schedule_timeout来睡眠指定时间,实际上这个时间设置的无限等待,也就是只能等着被唤醒,当前任务才能继续运行;
  • 调用up时,如果等待列表为空,表明没有多余的任务在等待信号量,直接将sem->count自加即可。如果等待列表非空,表明有任务正在等待信号量,那就需要对等待列表中的第一个任务(等待时间最长)进行唤醒操作,并从等待列表中将需要被唤醒的任务进行删除操作;

扩展的接口:

extern int __must_check down_interruptible(struct semaphore *sem);
extern int __must_check down_killable(struct semaphore *sem);
extern int __must_check down_trylock(struct semaphore *sem);
extern int __must_check down_timeout(struct semaphore *sem, long jiffies);

3.3 信号量缺点

对比 Mutex 锁,Semaphore 与Mutex 在实现上有一个重大的区别:ownership.

Mutex 被持有后有一个明确的 owner,而Semaphore 并没有owner,当一个进程阻塞在某个信号量上时,它没法知道自己阻塞在哪个进程(线程)之上。

没有ownership会带来以下几个问题:

  • 在保护临界区的时候,无法进行优先级反转的处理;
  • 系统无法对其进行跟踪断言处理,比如死锁检测等;
  • 信号量的调试变得更加麻烦;

因此,在Mutex 能满足要求的情况下,优先使用Mutex

参考:

https://www.cnblogs.com/LoyenWang/p/12907230.html

4. 读写信号量(rwsem)

读写信号量的原理与读写锁类似,读写信号量归根到底是 “信号量”,读写锁归根到底是 “自旋锁”,而信号量与自旋锁的区别一个可以睡眠,一个只能自旋。

读写信号量原理:

  • 允许多个读者同时进入临界区;
  • 读者与写者不能同时进入临界区(读者与写者互斥);
  • 写者与写者不能同时进入临界区(写者与写者互斥);
include/linux/rwsem.h

struct rw_semaphore {
        atomic_long_t count;               //读写信号量的计数

        atomic_long_t owner;               //当写者成功获取锁时,owner会指向锁的持有者
#ifdef CONFIG_RWSEM_SPIN_ON_OWNER
        struct optimistic_spin_queue osq;  //乐观自旋
#endif
        raw_spinlock_t wait_lock;          //自旋锁,用于count值的互斥访问
        struct list_head wait_list;        //不能立即获取到信号量的访问者,都会加到等待队列中
        ...
};

4.1 count

 * On 64-bit architectures, the bit definitions of the count are:
 *
 * Bit  0    - writer locked bit
 * Bit  1    - waiters present bit
 * Bit  2    - lock handoff bit
 * Bits 3-7  - reserved
 * Bits 8-62 - 55-bit reader count
 * Bit  63   - read fail bit
 *
 * On 32-bit architectures, the bit definitions of the count are:
 *
 * Bit  0    - writer locked bit
 * Bit  1    - waiters present bit
 * Bit  2    - lock handoff bit
 * Bits 3-7  - reserved
 * Bits 8-30 - 23-bit reader count
 * Bit  31   - read fail bit
 */
#define RWSEM_WRITER_LOCKED       (1UL << 0)
#define RWSEM_FLAG_WAITERS        (1UL << 1)
#define RWSEM_FLAG_HANDOFF        (1UL << 2)
#define RWSEM_FLAG_READFAIL       (1UL << (BITS_PER_LONG - 1))

#define RWSEM_READER_SHIFT        8
#define RWSEM_READER_BIAS         (1UL << RWSEM_READER_SHIFT)
#define RWSEM_READER_MASK         (~(RWSEM_READER_BIAS - 1))
#define RWSEM_WRITER_MASK         RWSEM_WRITER_LOCKED
#define RWSEM_LOCK_MASK           (RWSEM_WRITER_MASK|RWSEM_READER_MASK)
#define RWSEM_READ_FAILED_MASK    (RWSEM_WRITER_MASK|RWSEM_FLAG_WAITERS|\
                                        RWSEM_FLAG_HANDOFF|RWSEM_FLAG_READFAIL)

注意bit1:RWSEM_FLAG_WAITERS,当前持锁的如果是reader,来

4.3 osq(乐观自旋队列)

struct optimistic_spin_node {
        struct optimistic_spin_node *next, *prev;
        int locked; /* 1 if lock acquired */
        int cpu; /* encoded CPU # + 1 value */
};

struct optimistic_spin_queue {
        /*
         * Stores an encoded value of the CPU # of the tail node in the queue.
         * If the queue is empty, then it's set to OSQ_UNLOCKED_VAL.
         */
        atomic_t tail;
};

Optimistic spin queue 翻译过来为乐观自旋队列,即形成一组处于自旋状态的任务队列。

和等待队列不同,osq 中的任务都是当前正在执行的任务。osq 并没有将这些任务的 task struct 形成队列结构,而是把 per-cpu 的MSC lock 对象串联形成队列。如下图:

虽然都是自旋,但是自旋方式并不一样。Osq队列中的头部节点是持有osq锁的,只有该任务处于对rwsem的owner进行乐观自旋的状态(我们称之rwsem乐观自旋)。Osq队列中的其他节点都是自旋在自己的mcs lock上(我们称之mcs乐观自旋)。当头部的mcs lock释放掉后(结束rwsem乐观自旋),它会将mcs lock传递给下一个节点,从而让spinner队列上的任务一个个的按顺序进入rwsem的乐观自旋,从而避免了cache-line bouncing带来的性能开销。

cache-line bouncing的理解:为了以较低的成本大幅提升性能,现代CPU都有cache。cpu cache已经发展到了三级缓存结构。其中L1和L2cache为每一个核独有,L3则全部核共享。为了保证全部的核看到正确的内存数据,一个核在写入本身的L1 cache后,CPU会执行Cache一致性算法把对应的cacheline同步到其余核,这个过程并不很快。当多个cpu上频繁修改某个字段时,这个字段所在的cacheline被不停地同步到其它的核上,就像在核间弹来弹去,这个现象就叫作cache bouncing。

4.4 读写信号量初始化

4.4.1 静态初始化

include/linux/rwsem.h

#define __RWSEM_INITIALIZER(name)                                  \
        { __RWSEM_COUNT_INIT(name),                                \
          .owner = ATOMIC_LONG_INIT(0),                            \
          __RWSEM_OPT_INIT(name)                                   \
          .wait_lock = __RAW_SPIN_LOCK_UNLOCKED(name.wait_lock),   \
          .wait_list = LIST_HEAD_INIT((name).wait_list),           \
          __RWSEM_DEBUG_INIT(name)                                 \
          __RWSEM_DEP_MAP_INIT(name) }

#define DECLARE_RWSEM(name) \
        struct rw_semaphore name = __RWSEM_INITIALIZER(name)

4.4.2 动态初始化

include/linux/rwsem.h

#define init_rwsem(sem)                                        \
do {                                                           \
        static struct lock_class_key __key;                    \
                                                               \
        __init_rwsem((sem), #sem, &__key);                     \
} while (0)
kernel/locking/rwsem.c

void __init_rwsem(struct rw_semaphore *sem, const char *name,
                  struct lock_class_key *key)
{
        atomic_long_set(&sem->count, RWSEM_UNLOCKED_VALUE);  //初始化count
        raw_spin_lock_init(&sem->wait_lock);                 //初始化wait_lock
        INIT_LIST_HEAD(&sem->wait_list);                     //初始化wait_list
        atomic_long_set(&sem->owner, 0L);                    //初始化owner
#ifdef CONFIG_RWSEM_SPIN_ON_OWNER
        osq_lock_init(&sem->osq);                            //初始化osq
#endif
}
EXPORT_SYMBOL(__init_rwsem);

1.2 down_read()

1.3 up_read()

1.4 down_write()

1.5 up_write()

参考:https://zhuanlan.zhihu.com/p/553482378

5. 自旋锁(spinlock)

spinlock 是一种不可休眠锁,spinlock在持锁失败后,不会进行睡眠,而是自旋等待。

线程睡眠、唤醒时都需要进行调度,这部分线程上、下文的切换也是性能开销。而spinlock则在持锁失败后,不会进行睡眠,少了这一部分的开销。

spinlock不适合保护很大的临界区,因为在持锁后会关闭抢占或中断,如果持锁时间过长,持锁线程以及持锁未成功进行自旋线程所在cpu会出现调度不及时带来的性能问题。 

另外,在软、硬中断上下文,是不允许睡眠的,所以 mutex不能在这里使用,需要使用spinlock。

5.1 自旋锁与UP、SMP 的关系

UP:Unified Processor

SMP:Symmetric Multi-Processors

根据自旋锁的逻辑,自旋锁的临界区是不能休眠的。在UP下,只有一个CPU,如果我们执行到了临界区,此时自旋锁是不可能处于加锁状态的。因为我们正在占用CPU,又没有其它的CPU,其它的临界区要么没有到来、要么已经执行过去了。所以我们是一定能获得自旋锁的,所以自旋锁对UP来说是没有意义的。但是为了在UP和SMP下代码的一致性,UP下也有自旋锁,但是自旋锁的定义就变成了空结构体,自旋锁的加锁操作就退化成了禁用抢占,自旋锁的解锁操作也就退化成了开启抢占。所以说自旋锁只适用于SMP,但是在UP下也提供了兼容操作。

5.2 自旋锁的发展

5.2.1 wild spinlock(原始自旋锁)

struct spinlock {
    int locked;
};

void spin_lock(struct spinlock *lock)
{
    while (lock->locked);
    lock->locked = 1;
}

void spin_unlock(struct spinlock *lock)
{
    lock->locked = 0;
}

锁的持有者会将locked 置1,释放时会将locked 置0,其他不能获得 spinlock 的进程原地自旋。

wild spinlock 的实现非常简单,但这种简单的背后却掩盖着血雨腥风的竞争,一旦spinlock 被释放,哪个 CPU 的cache 先看到 locked 的值,就优先获得 spinlock,没有排队的机制,这样导致某 CPU 可能长期等锁,从而带来延迟不确定问题,甚至导致CPU 饿死。

为了改善获取lock 乱插队,ticket spinlock 出现了。

5.2.2 ticket spinlock(票号自旋锁)

struct spinlock {
        unsigned short owner;
        unsigned short next;
};

ticket spinlock引入排队机制,以FIFO的顺序处理spinlock的申请,第一个CPU抢占spinlock时会先获取到spinlock,后面的CPU根据他们抢占的先后顺序来获取到spinlock,这就实现了获取spinlock的公平性。

每一个参与获取 spinlock 的CPU 的cache 都会缓存两个变量:current ticketnext ticket。每当有 CPU 要获取锁时,会向外发出一个 ticket,即 next ticket 加 1,当有 CPU 要释放锁时,current ticket 加1,每个 CPU 发现自己的ticket 和 current ticket 相等时就获得到锁。

所有试图获取spinlock的 CPU 都在不停的读取spinlock中两个变量的数值变化,当spinlock的值被更改时,这些CPU对应的cache line都会被invalidate,它们必须重新从内存读取新的spinlock的值到自己的cache line中。

而事实上,只有队列中最先排队的那个CPU的cache line被invalidate才是有意义的,对于其他的CPU来说,自己的cache line被invalidate完全是在浪费总线资源。

实际上wild spinlock也有这种浪费总线的问题。为了解决这种资源浪费问题,MCS spinlock出现了。

5.2.3 MCS spinlock(MCS自旋锁)

MCS 锁机制是由John Mellor CrummeyMichael Scott在论文中《algorithms for scalable synchronization on shared-memory multiprocessors》提出的,并以他俩的名字来命名;

struct mcs_spinlock
{
        struct mcs_spinlock *next;
        int locked;
};

lock 操作值需要将所有所属 CPU 的 mcs_spinlock 结构体串联成到链表尾部,然后自旋直到自己的 mcs_spinlock 的locked 成员被置1.

unlock 的操作只需要将解锁的 CPU 对应的 mcs_spinlock 结构体的next 的locked 成员置 1,交出锁的使用权,即通知链表中下一个 CPU 可以获取锁了。

5.2.4 qspinlock(队列自旋锁)

include/asm-generic/qspinlock_types.h

typedef struct qspinlock {
        union {
                atomic_t val;

#ifdef __LITTLE_ENDIAN
                struct {
                        u8        locked;
                        u8        pending;
                };
                struct {
                        u16        locked_pending;
                        u16        tail;
                };
#else
                ...
#endif
        };
} arch_spinlock_t;
  • locked:一个字节,表示这个锁是否被持有;
  • pending:一个字节,当锁 unlock 之后,持有这个位的 CPU 最先持锁;
  • locked_pending:双字节,对应 locked、pending;
  • tail:双字节,分两部分编码:
    • 2 位:用来编码执行流(线程、软中断、硬中端、非屏蔽中断)
    • 14位:用来编码CPU index

qspinlock 的数据结构大致分布如下:

正如qspinlock的作者所说:这个补丁可能不是锁竞争问题的终极解决方案。但重要的是,qspinlock在竞争不严重的场景表现出色,释放一个ticket 自旋锁需要执行一个“读-改-写”原子指令,而qspinlock的释放只需一个写操作。

5.3 自旋锁初始化

5.3.1 动态初始化

include/linux/spinlock.h

# define spin_lock_init(_lock)                        \
do {                                                  \
        spinlock_check(_lock);                        \
        *(_lock) = __SPIN_LOCK_UNLOCKED(_lock);       \
} while (0)
include/linux/spinlock_types.h

#define ___SPIN_LOCK_INITIALIZER(lockname)           \
        {                                            \
        .raw_lock = __ARCH_SPIN_LOCK_UNLOCKED,       \
        ...

#define __SPIN_LOCK_INITIALIZER(lockname) \
        { { .rlock = ___SPIN_LOCK_INITIALIZER(lockname) } }

5.3.2 静态初始化

include/linux/spinlock_types.h

#define __SPIN_LOCK_UNLOCKED(lockname) \
        (spinlock_t) __SPIN_LOCK_INITIALIZER(lockname)

#define DEFINE_SPINLOCK(x)        spinlock_t x = __SPIN_LOCK_UNLOCKED(x)

5.3.3 实际初始化

include/asm-generic/qspinlock_types.h

#define        __ARCH_SPIN_LOCK_UNLOCKED        { { .val = ATOMIC_INIT(0) } }

无论是动态初始化,还是静态初始化,最终就是调用__ARCH_SPIN_LOCK_UNLOCKED将qspinlock 的成员都置 0.

5.4 spin_lock()

include/linux/spinlock.h

static __always_inline void spin_lock(spinlock_t *lock)
{
        raw_spin_lock(&lock->rlock);
}

#define raw_spin_lock(lock)        _raw_spin_lock(lock)

include/asm-genric/qspinlock.h

/**
 * spin_lock()的核心处理函数
 * 有两个宏需要关注:_Q_LOCKED_VAL(1<<0) 和 _Q_PENDING_VAL(1<<8)
 * 
 */
static __always_inline void queued_spin_lock(struct qspinlock *lock)
{
        int val = 0;

        if (likely(atomic_try_cmpxchg_acquire(&lock->val, &val, _Q_LOCKED_VAL)))
                return;

        queued_spin_lock_slowpath(lock, val);
}

这里可快速上锁慢速上锁,当lock->val 为0 的时候,认为没有线程持有该锁,可以快速上锁,否则进入慢速。

慢速上锁这里暂不剖析,详细可以查看源码。

5.5 spin_unlock()

include/linux/spinlock.h

static __always_inline void spin_unlock(spinlock_t *lock)
{
        raw_spin_unlock(&lock->rlock);
}

include/asm-genric/qspinlock.h

static __always_inline void queued_spin_unlock(struct qspinlock *lock)
{
        smp_store_release(&lock->locked, 0);
}

可以看到队列自旋锁的解锁确实很简单,只需把 locked 字节设为 0 就可以了。

参考:

https://zhuanlan.zhihu.com/p/506795930

https://zhuanlan.zhihu.com/p/551234849

https://zhuanlan.zhihu.com/p/648188138

6. 读写锁(queue read/write lock)

Linux 内核在 spinlock 基础上实现了 读/写 控制的锁rwlock_t,通过如下数据结构可以看出在 arch_spinlock_t 的基础上多加了一个原子变量cnts,也就是说读写锁的本质就是自旋锁

读写锁的原理:

  • 在没有写锁时,任意读锁都可以并发;
  • 当有写锁时,读锁或写锁必须自旋等待;
  • 当有读锁时,读锁可以并发(不阻塞),写锁自旋等待;
include/linux/rwlock_types.h

typedef struct {
        arch_rwlock_t raw_lock;
#ifdef CONFIG_DEBUG_SPINLOCK
        unsigned int magic, owner_cpu;
        void *owner;
#endif
#ifdef CONFIG_DEBUG_LOCK_ALLOC
        struct lockdep_map dep_map;
#endif
} rwlock_t;
include/asm-generic/qrwlock_types.h

typedef struct qrwlock {
        union {
                atomic_t cnts;
                struct {
#ifdef __LITTLE_ENDIAN
                        u8 wlocked;        /* Locked for write? */
                        u8 __lstate[3];
#else
                        ...
#endif
                };
        };
        arch_spinlock_t                wait_lock;
} arch_rwlock_t;

读写锁中用一个原子变量cnts用以统计读写状态:

  • 高 24 位用以读锁操作计数,读锁操作成功计数+1;
  • 低 8 位用以记录写锁状态:有写意愿、写锁使用中、没有写操作;

6.1 读写锁初始化

include/linux/rwlock.h

# define rwlock_init(lock)                                        \
        do { *(lock) = __RW_LOCK_UNLOCKED(lock); } while (0)
include/linux/rwlock_types.h

#define __RW_LOCK_UNLOCKED(lockname) \
        (rwlock_t)        {        .raw_lock = __ARCH_RW_LOCK_UNLOCKED,        \
                                RW_DEP_MAP_INIT(lockname) }
include/asm-generic/qrwlock_types.h

#define        __ARCH_RW_LOCK_UNLOCKED {                \
        { .cnts = ATOMIC_INIT(0), },                    \
        .wait_lock = __ARCH_SPIN_LOCK_UNLOCKED,         \
}

初始化就是将cnts 置0,并初始化自旋锁。

6.2 read_lock()

include/linux/rwlock.h

#define read_lock(lock)            _raw_read_lock(lock)

include/asm-generic/qrwlock.h

static inline void queued_read_lock(struct qrwlock *lock)
{
        int cnts;

        cnts = atomic_add_return_acquire(_QR_BIAS, &lock->cnts);
        if (likely(!(cnts & _QW_WMASK)))
                return;

        /* The slowpath will decrement the reader count, if necessary. */
        queued_read_lock_slowpath(lock);
}

读锁上锁分快速上锁慢速上锁,当没有写锁的时候,read_lock() 可以成功返回,而当有写锁时,则进入慢速,具体逻辑可以查看源码。

6.3 read_unlock()

include/linux/rwlock.h

#define read_unlock(lock)                _raw_read_unlock(lock)

include/asm-generic/qrwlock.h

static inline void queued_read_unlock(struct qrwlock *lock)
{
        (void)atomic_sub_return_release(_QR_BIAS, &lock->cnts);
}

可以看到读锁的解锁确实很简单,只需把对读标记变量 __lstate 进行 -1 操作。

6.4 write_lock()

include/linux/rwlock.h

#define write_lock(lock)        _raw_write_lock(lock)

include/am-generic/qrwlock.h

static inline void queued_write_lock(struct qrwlock *lock)
{
        int cnts = 0;
        /* Optimize for the unfair lock case where the fair flag is 0. */
        if (likely(atomic_try_cmpxchg_acquire(&lock->cnts, &cnts, _QW_LOCKED)))
                return;

        queued_write_lock_slowpath(lock);
}

写锁上锁时,会优先尝试确认lock->cnts 时候有值,如果没有则直接获取写锁,并成功返回。

如果已经有锁,则调用queued_write_lock_slowpath进入慢速上锁。

6.5 write_unlock()

include/linux/rwlock.h

#define write_unlock(lock)                _raw_write_unlock(lock)

include/asm-generic/qrwlock.h

static inline void queued_write_unlock(struct qrwlock *lock)
{
        smp_store_release(&lock->wlocked, 0);
}

可以看到,写锁解锁时很简单,只需把 wlocked 字节设为 0 就可以了。

6.6 读写锁缺点

read_lock 与 write_lock 具有相同的优先权:读者必须等所有的写完成,写者必须等待读操作完成。

当接二连三都是读者操作,写锁可能会一直等待,这样造成写者饿死的情况。

7. 顺序锁

从下面定义可知,顺序锁本质还是自旋锁,只不过需要携带 seqcount。

include/linux/seqlock.h

typedef struct {
        seqcount_spinlock_t seqcount;
        spinlock_t lock;
} seqlock_t;

typedef struct seqcount_spinlock {
        seqcount_t        seqcount;
#if defined(CONFIG_LOCKDEP) || defined(CONFIG_PREEMPT_RT)
        spinlock_t        *lock;
#endif
} seqcount_spinlock_t;

typedef struct seqcount {
        unsigned sequence;
#ifdef CONFIG_DEBUG_LOCK_ALLOC
        struct lockdep_map dep_map;
#endif
} seqcount_t;

顺序锁提供了一种方式:读者不阻塞写,与读写锁相比,不会出现写者饿死的情况。

这种方式下,读者不会阻塞写者,读者在读数据的时候,写者可以写数据。顺序锁有序列号,写者把序列号加 1,如果读者检测到序列号有变化,发现写者修改了数据,将会重试,读者的开销可能比较高

7.1 顺序锁初始化

7.1.1 动态初始化

include/linux/seqlock.h

#define seqlock_init(sl)                                                    \
        do {                                                                \
                spin_lock_init(&(sl)->lock);                                \
                seqcount_spinlock_init(&(sl)->seqcount, &(sl)->lock);       \
        } while (0)

结构体中变量 lock 通过spin_lock_init进行自旋锁初始化,详细看第 5.3.1 节。

include/linux/seqlock.h

#define seqcount_spinlock_init(s, lock)           seqcount_LOCKNAME_init(s, lock, spinlock)

#define seqcount_LOCKNAME_init(s, _lock, lockname)                    \
        do {                                                          \
                seqcount_##lockname##_t *____s = (s);                 \
                seqcount_init(&____s->seqcount);                      \
                __SEQ_LOCK(____s->lock = (_lock));                    \
        } while (0)

# define seqcount_init(s) __seqcount_init(s, NULL, NULL)

static inline void __seqcount_init(seqcount_t *s, const char *name,
                                          struct lock_class_key *key)
{
        lockdep_init_map(&s->dep_map, name, key, 0);
        s->sequence = 0;
}

seqcount_spinlock_init函数主要是将 sequence 置0.

7.1.2 静态初始化

include/linux/seqlock.h

#define DEFINE_SEQLOCK(sl) \
                seqlock_t sl = __SEQLOCK_UNLOCKED(sl)
                
#define __SEQLOCK_UNLOCKED(lockname)                                          \
        {                                                                     \
                .seqcount = SEQCNT_SPINLOCK_ZERO(lockname, &(lockname).lock), \
                .lock =        __SPIN_LOCK_UNLOCKED(lockname)                 \
        }

这里的自旋锁初始化使用__SPIN_LOCK_UNLOCKED宏,详细可以查看第 5.3.2 节。

另外,SEQCNT_SPINLOCK_ZERO目的也是初始化 sequence 为0.

7.2 读锁的三种方式

读者因为上锁的方式不同可以分:

  • 顺序读者

  • 持锁读者

7.2.1 read_seqbegin / read_seqretry

这种方式无加锁访问,读者在读临界区之前先读取sequence,退出临界区操作后再次读sequence并进行比较,如果发现前后的sequence不相等,则说明有写者更新内容,需要重新操作临界区。

所以,这种方式可能会给读者带来开销。

7.2.2 read_seqlock_excl / read_sequnlock_excl

这种方式就是利用seqlock 中的spinlock,进行自旋。

扩展接口有:

read_seqlock_excl_bh  / read_sequnlock_excl_bh   //申请自旋锁,禁止当前处理器的软中断
read_seqlock_excl_irq / read_sequnlock_excl_irq  //申请自旋锁,禁止当前处理器的硬中断
read_seqlock_excl_irqsave / read_sequnlock_excl_irqrestore  //申请自旋锁,保存当前cpu的硬中断状态,并禁止当前cpu的硬中断

7.2.3 read_seqbegin_or_lock

读者的另一种方式是read_seqbegin_or_lock,可以根据sequence是否为奇数来确定是否有写者操作。

通过下文写锁操作可知,当有写者申请seqlock 时会将sequence进行+1操作,这样读者根据sequence是否为奇数确认是否进行写操作。

扩展接口:

read_seqbegin_or_lock_irqsave / done_seqretry_irqrestore
//如果没有写者,那么读者成为顺序读者;
//如果写着正在写数据,读者成为持锁读者,申请自旋锁,保存当前处理器硬中断状态,并禁止当前处理器的硬中断

7.3 write_seqlock / write_sequnlock

顺序锁的写锁只有一种方式,上锁时先是spin_lock,然后将sequence进行自增处理;解锁时先是spin_unlock,然后将sequence进行自增处理。

扩展接口:

write_seqlock_bh  / write_sequnlock_bh                 //申请写锁,并禁止当前cpu的软中断
write_seqlock_irq / write_sequnlock_irq                //申请写锁,并禁止当前cpu的硬中断
__write_seqlock_irqsave / write_sequnlock_irqrestore   //申请写锁,保存当前cpu的硬中断状态,禁止当前cpu的硬中断

流程中的 smp_rmb / smp_wmb两个函数是内存屏障操作,作用是告诉编译器内存中的值已经改变,之前对内存的缓存(缓存到寄存器) 都需要抛弃,屏障之后的内存操作需要重新从内存 load,而不能使用之前的寄存器缓存的值。

内存屏障就像是代码中一道不可逾越的屏障,屏障之前的 load/store 指令不能跑到屏障后边,同理,后边的也不能跑到前边。

7.4 顺序锁缺点

如果读者采用方式1,临界区中存在地址(指针) 操作,如果写者把地址修改了,那就可能造成访问错误。

8. RCU

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2038651.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

vue3结合海康WEB开发包,开发web在线预览视频

我们这里选择V3.3版本 文档地址&#xff1a;https://open.hikvision.com/download/5cda567cf47ae80dd41a54b3?type20&id4c945d18fa5f49638ce517ec32e24e24 解压过后&#xff0c;会有三个文件夹 在docs中&#xff0c;点开Demo使用说明&#xff0c;按照流程先测试下&…

赋能基层,融合创新:EasyCVR视频汇聚平台构建平安城市视频共享系统

一、雪亮工程建设的意义 雪亮工程的核心在于通过高清视频监控、环境监测和智能预警等先进技术手段&#xff0c;构建一个高效、智能、安全、便捷的社会安全防控体系。这一工程的建设不仅代表了现代化科技手段在城市治安管理中的应用&#xff0c;更是提升社会安全保障能力、推动…

树形结构查找(B树、B+树)

平衡树结构的树高为 O(logn) &#xff0c;平衡树结构包括两种平衡二叉树结构&#xff08;分别为 AVL 树和 RBT&#xff09;以及一种树结构&#xff08;B-Tree&#xff0c;又称 B 树&#xff0c;它的度大于 2 &#xff09;。AVL 树和 RBT 适合内部存储的应用&#xff0c;而 B 树…

CompreFace Study

系列文章目录 第一章 CompreFace Installation 第二章 Face verification POC 文章目录 系列文章目录前言一、What is the ComreFace&#xff1f;二、How to install the CompreFace? 1.On Linux for CompreFace 1.2.02.Troubleshooting总结 前言 此文旨在记录学习CompreF…

萤石取流播放失败自助排障及常见错误码解决方案

一、在使用播放地址播放时遇到播放失败的情况&#xff0c;可使用排障工具排查具体原因&#xff0c;以下具体介绍排障工具的使用方法 1、在浏览器里打开排障工具&#xff0c;地址&#xff1a;萤石开放平台-提供持续稳定的以音视频为主的全场景、多功能综合性服务 2、在第一行输入…

安全无忧!Windows7全补丁旗舰版:集成所有补丁!

今日&#xff0c;系统之家小编给大家分享集成所有补丁的Windows7旗舰版系统&#xff0c;集成至2023.12所有官方补丁&#xff0c;修复了系统高危漏洞&#xff0c;让大家时刻都能舒心地展开操作。系统基于微软 Windows 7 2009 SP1 旗舰版进行离线制作&#xff0c;全新升级的优化方…

本地环境VMware使用代理解决 Docker 镜像拉取问题

引言 本文将分享我在 Windows 10 环境下&#xff0c;通过 VMware 运行的 CentOS 7.8 虚拟机中配置 Docker 代理&#xff0c;成功解决了镜像拉取问题的经验。 问题描述 在尝试启动一个依赖 Docker 的 GitHub 项目时&#xff0c;拉取 Docker 镜像的失败。尝试配置了几个国内源…

(附源码)基于springboot的智慧社区管理系统-计算机毕设 06797

基于springboot的智慧社区管理系统 摘 要 SpringBoot智慧社区管理系统是一款基于SpringBoot框架开发的智能化社区管理软件&#xff0c;致力于提升社区管理效率和服务质量。该系统涵盖了社区入住管理、物业费管理、公共设施预约等功能&#xff0c;支持在线报修、信息发布、社区活…

Java语言程序设计——篇十三(2)

&#x1f33f;&#x1f33f;&#x1f33f;跟随博主脚步&#xff0c;从这里开始→博主主页&#x1f33f;&#x1f33f;&#x1f33f; 欢迎大家&#xff1a;这里是我的学习笔记、总结知识的地方&#xff0c;喜欢的话请三连&#xff0c;有问题可以私信&#x1f333;&#x1f333;&…

【运维】报错Resource averaged_perceptron_tagger_eng not found.

文章目录 报错信息解决报错信息 able of handling various complex tasks. Please report the progress of this project to the team members.> ========================<

回溯算法探索篇Ⅲ

Leetcode93——复原IP地址 题目描述&#xff1a; 有效 IP 地址 正好由四个整数&#xff08;每个整数位于 0 到 255 之间组成&#xff0c;且不能含有前导 0&#xff09;&#xff0c;整数之间用 . 分隔。 例如&#xff1a;"0.1.2.201" 和 "192.168.1.1" 是…

PyTorch — 初学者教程

一、说明 在本文中,我将编译 PyTorch 的初学者教程。本教程大量使用了官方 PyTorch 教程中的材料。

Java SE--IO流

一.File类型 如果我们想在程序中操作或者描述一个文件夹或文件&#xff0c;可以使用File类型 File类型在java.io包下 File可以新建&#xff0c;删除&#xff0c;移动&#xff0c;修改&#xff0c;重命名文件夹&#xff0c;也可以对文件或者文件夹的属性进行访问&#xff1b;…

嵌入式软件--模电基础 DAY 2

强电和弱电&#xff0c;简单一点是以电死人为标准的&#xff0c;交流电36伏特以下&#xff0c;直流电24V以下&#xff0c;为安全电压&#xff0c;是为弱电&#xff0c;反则强电。 市电进入家庭&#xff0c;连接你的电脑&#xff0c;220V的电压为什么没有让你感到危险&#xff…

【TiDB】09-修改tidb客户端访问密码

目录 1、修改配置文件 2、停止tidb-server 3、以root方式启动脚本 4、修改密码 5、停止脚本重启服务 1、修改配置文件 进入tidb-server默认部署位置 #切换tidb账号 su tidb# 进入tidb-server部署路径 cd /tidb-deploy/tidb-4000# 修改配置 vim ./conf/tidb.toml添加内容…

Nginx之我不会的

安装 windows下安装 这里我们选择windows下的稳定版本 之后就是解压&#xff0c;安装到英文目录下 启动&#xff08;关闭&#xff09;命令 windows命令 启动 nginx.exe 或者 start nginx 关闭 nginx -s quit&#xff08;会有一定程序退出&#xff09; 或 nginx -s stop &…

企业低代码平台那个好用?企业低代码推荐

近些年&#xff0c;很多大型的互联网公司开始了自研低代码开发平台的尝试&#xff0c;包括阿里、腾讯、华为等等国内头部大厂。由此我们可以了解到即使是开发能力富余的互联网大厂&#xff0c;也同样需要低代码的快速响应需求能力来应对数字化时代多变的功能需求。然而&#xf…

本地phpstudy部署算命系统,用户端是H5页面,支持微信支付宝支付,支持微信支付宝登录

算命系统本地部署教程 0. 技术架构1. 启动Apache、MySQL服务2. 创建前台和后台两个网站3. Navicat连接数据库4. 登录后台是长这个样子5. 登录前台登录样子6. 代码结构是 0. 技术架构 前端&#xff1a;HTMLCSSJquery 后端&#xff1a;PHP 数据库&#xff1a;MySQL 1. 启动Ap…

C++简单实现多态案例-制作饮品

代码示例如下&#xff1a; #include<iostream> using namespace std;class AbstractDrinking { public:virtual void Soil() 0;//第一步、煮开水virtual void Brew() 0;//第二步、冲泡virtual void PourInCup() 0;//第三步、倒入杯中virtual void AddSomething() 0;…

Lab 1 实验 MapReduce

&#x1f442; 若月亮没来 (若是月亮还没来)&#xff08;若是月亮还没来&#xff09; - 王宇宙Leto/乔浚丞 - 单曲 - 网易云音乐 目录 &#x1f33c;参考代码 &#x1f419;解析 &#x1f41f;mrsequential.go &#x1f41f;mrapps/wc.go &#x1f4d5;实验--准备 &…