Linux中读写锁--写锁优先

本文详细探讨了Linux操作系统中的读写锁机制,重点解析了写锁优先的策略。通过示例代码`test.c`,阐述了如何在实际编程中应用这一概念,帮助读者理解读写锁在并发控制中的作用及其优势。
摘要由CSDN通过智能技术生成

my_pthread_rwlock.c

#include"my_pthread_rwlock.h"
#include<errno.h>

int my_pthread_rwlock_rdlock(my_pthread_rwlock_t *rw)
{
    int result;
    if(rw->rw_magic != RW_MAGIC)
        return -1;
    if((result = pthread_mutex_lock(&rw->rw_mutex)) != 0)
        return result;

    while(rw->rw_refcount<0 || rw->rw_nwaitwriters>0)
    {
        rw->rw_nwaitreaders++;
        result = pthread_cond_wait(&rw->rw_condreaders, &rw->rw_mutex);
        rw->rw_nwaitreaders--;
        if(result != 0)
            break;
    }
    if(result == 0)
        rw->rw_refcount++;
    pthread_mutex_unlock(&rw->rw_mutex);
    return result;
}

int my_pthread_rwlock_wrlock(my_pthread_rwlock_t *rw)
{
    int result;
    if(rw->rw_magic != RW_MAGIC)
        return -1;

    if((result = pthread_mutex_lock(&rw->rw_mutex)) != 0)
        return result;

    while(rw->rw_refcount != 0)
    {
        rw->rw_nwaitwriters++;
        result = pthread_cond_wait(&rw->rw_condwriters, &rw->rw_mutex);
        rw->rw_nwaitwriters--;
        if(result != 0)
            break;
    }
    if(result == 0)
        rw->rw_refcount = -1;

    pthread_mutex_unlock(&rw->rw_mutex);
    return result;
}

int my_pthread_rwlock_unlock(my_pthread_rwlock_t *rw)
{
    int result;
    if(rw->rw_magic != RW_MAGIC)
        return -1;
    if((result = pthread_mutex_lock(&rw->rw_mutex)) != 0)
        return result;

    if(rw->rw_refcount > 0)
        rw->rw_refcount--;
    else if(rw->rw_refcount == -1)
        rw->rw_refcount = 0;
    else
        printf("unlock error.\n");

    if(rw->rw_nwaitwriters > 0)
    {
        if(rw->rw_refcount == 0)
        {
            result = pthread_cond_signal(&rw->rw_condwriters);
        }
    }
    else if(rw->rw_nwaitreaders > 0)
        result = pthread_cond_broadcast(&rw->rw_condreaders);

    pthread_mutex_unlock(&rw->rw_mutex);
    return result;
}

int my_phread_rwlock_destroy(my_pthread_rwlock_t *rw)
{
    if(rw->rw_magic != RW_MAGIC)
        return (EINVAL);
    if(rw->rw_refcount != 0 || rw->rw_nwaitreaders != 0 || rw->rw_nwaitwriters != 0)
        return (EBUSY);
    pthread_mutex_destroy(&rw->rw_mutex);
    pthread_cond_destroy(&rw->rw_condreaders);
    pthread_cond_destroy(&rw->rw_condwriters);
    rw->rw_magic = 0;
    return 0;
}

my_pthread_rwlock_tryrdlock(my_pthread_rwlock_t *rw)
{
    int result;
    if(rw->rw_magic != RW_MAGIC)
        return (EINVAL);
    if((result = pthread_mutex_lock(&rw->rw_mutex)) != 0)
        return result;
    if(rw->rw_refcount <0 || rw->rw_nwaitwriters > 0)
        result = (EBUSY);
    else
        rw->rw_refcount++;
    pthread_mutex_unlock(&rw->rw_mutex);
    return result;
}

my_pthread_rwlock_trywrlock(my_pthread_rwlock_t *rw)
{
    int result;
    if(rw->rw_magic != RW_MAGIC)
        return (EINVAL);
    if((result = pthread_mutex_lock(&rw->rw_mutex)) != 0)
        return result;
    if(rw->rw_refcount != 0)
        return (EBUSY);
    else
        rw->rw_refcount = -1;
    pthread_mutex_unlock(&rw->rw_mutex);
    return result;
}




my_pthread_rwlock.h

#pragma once

#include<pthread.h>
#include<stdio.h>

typedef struct
{
    pthread_mutex_t rw_mutex;
    pthread_cond_t  rw_condreaders;
    pthread_cond_t  rw_condwriters;
    int             rw_magic;
    int             rw_nwaitreaders;
    int             rw_nwaitwriters;
    int             rw_refcount;       // 0 >0 ==-1
}my_pthread_rwlock_t;

#define  RW_MAGIC  0x20180326

#define MY_PTHREAD_RWLOCK_INITIALIZER {PTHREAD_MUTEX_INITIALIZER, PTHREAD_COND_INITIALIZER, PTHREAD_COND_INITIALIZER,\
RW_MAGIC,0,0,0}

typedef int  my_pthread_rwlockattr_t;


int my_pthread_rwlock_init(my_pthread_rwlock_t *rw, my_pthread_rwlockattr_t *attr);
int my_pthread_rwlock_destroy(my_pthread_rwlock_t *rw);
int my_pthread_rwlock_rdlock(my_pthread_rwlock_t *rw);
int my_pthread_rwlock_wrlock(my_pthread_rwlock_t *rw);
int my_pthread_rwlock_unlock(my_pthread_rwlock_t *rw);
int my_pthread_rwlock_tryrdlock(my_pthread_rwlock_t *rw);
int my_pthread_rwlock_trywrlock(my_pthread_rwlock_t *rw);

test.c//测试文件

#include<stdio.h>
#include<unistd.h>
#include<pthread.h>
#include"my_pthread_rwlock.h"
#include"my_pthread_rwlock.c"

my_pthread_rwlock_t rwlock = MY_PTHREAD_RWLOCK_INITIALIZER;

void * thread_fun1(void *arg)
{
    my_pthread_rwlock_wrlock(&rwlock);
    printf("thread 1 wrlock.\n");
    sleep(3);
    my_pthread_rwlock_unlock(&rwlock);
}
void * thread_fun2(void *arg)
{
      my_pthread_rwlock_wrlock(&rwlock);
    //my_pthread_rwlock_trywrlock(&rwlock);
    //printf("thread 2 trywrlock\n");
      printf("thread 2 wrlock.\n");
    my_pthread_rwlock_unlock(&rwlock);
}
void * thread_fun3(void *arg)
{
      my_pthread_rwlock_rdlock(&rwlock);
    //my_pthread_rwlock_tryrdlock(&rwlock);
    //printf("thread 3 tryrdlock.\n");
      printf("thread 3 rdlock.\n");
    my_pthread_rwlock_unlock(&rwlock);
}

int main()
{
    pthread_t tid1, tid2, tid3;
    pthread_create(&tid1, NULL, thread_fun1, NULL);
    sleep(1);
    pthread_create(&tid3, NULL, thread_fun3, NULL);
    pthread_create(&tid2, NULL, thread_fun2, NULL);

    pthread_join(tid1, NULL);
    pthread_join(tid2, NULL);
    pthread_join(tid3, NULL);

    return 0;
}

/*
pthread_rwlock_t rwlock = PTHREAD_RWLOCK_INITIALIZER;

void* thread_fun1(void *arg)
{
    pthread_rwlock_wrlock(&rwlock);
    printf(" thread fun 1 wrlock.\n");
    sleep(15);
    printf("thread fun 1 unlock.\n");
    pthread_rwlock_unlock(&rwlock);
}
void* thread_fun2(void *arg)

{
    int index = *(int*)arg;
    printf("thread fun 2[ %d] start up.\n",index);
    pthread_rwlock_wrlock(&rwlock);
    printf("thread fun 2:[%d] wrlock\n",index);
    pthread_rwlock_unlock(&rwlock);
}
void* thread_fun3(void *arg)
{
    int index = *(int *)arg;
    printf("thread fun 2[ %d] start up.\n",index);
    pthread_rwlock_rdlock(&rwlock);
    printf("thread fun 3:[%d] rdlock\n",index);
    pthread_rwlock_unlock(&rwlock);
}

int main()
{
    pthread_t tid1, tid2[5], tid3[5];
    
    pthread_create(&tid1, NULL, thread_fun1, NULL);
    sleep(1);
    int i;
    for(i=0; i<5; ++i)
    {
        pthread_create(&tid2[i], NULL, thread_fun2, &i);
        sleep(1);
    }
    for(i=0; i<5; ++i)
    {
        pthread_create(&tid3[i], NULL, thread_fun3, &i);
    }

    for(i=0; i<5; ++i)
    {
        pthread_join(tid2[i], NULL);
        pthread_join(tid3[i], NULL);
    }
    pthread_join(tid1, NULL);
    return 0;
}

/*
void* thread_fun1(void *arg)
{
    pthread_mutex_lock(&mutex);
    printf("This is thread fun 1.\n");

    sleep(3);
    printf("thread fun 1 cond wait.......\n");
    pthread_cond_wait(&cond, &mutex);
    printf("thread fun 1 waked up.\n");

    pthread_mutex_unlock(&mutex);
}
void* thread_fun2(void *arg)
{
    printf("This is thread fun 2.\n");
    pthread_mutex_lock(&mutex);
    printf("aaaaaaaaaaaaaaa\n");
    sleep(2);
    printf("cond signal thread......\n");
    pthread_cond_signal(&cond);
    sleep(3);
    printf("====================\n");
    pthread_mutex_unlock(&mutex);
}
void * thread_fun3(void *arg)
{
    printf("This is thread fun 3.\n");
    pthread_mutex_lock(&mutex);
    printf("bbbbbbbbbbbbbbbbb\n");
    pthread_cond_signal(&cond);
    pthread_mutex_unlock(&mutex);
}

int main()
{
    pthread_t tid1, tid2, tid3;
    pthread_create(&tid1, NULL, thread_fun1, NULL);
    sleep(1);
    pthread_create(&tid3, NULL, thread_fun3, NULL);
    pthread_create(&tid2, NULL, thread_fun2, NULL);

    pthread_join(tid1, NULL);
    pthread_join(tid2, NULL);
    return 0;
}
*/


读写锁是用于多线程环境下保护共享资源的一种机制。在 Linux 内核读写锁的实现是基于 spinlock 和原子操作的。下面我将简单介绍一下 Linux 内核读写锁的源码实现。 读写锁的定义: ```c typedef struct { raw_rwlock_t raw_lock; } rwlock_t; ``` 其,raw_rwlock_t 是一个原始读写锁类型,它是由内核提供的一个结构体类型,定义在 include/linux/spinlock_types.h 文件。 raw_rwlock_t 的定义: ```c typedef struct { arch_rwlock_t raw_lock; } raw_rwlock_t; typedef struct { unsigned int lock; } arch_rwlock_t; ``` 其,arch_rwlock_t 是一个体系结构相关的原始读写锁类型,它的实现由不同的处理器架构提供。 下面是 x86_64 架构下的 arch_rwlock_t 实现: ```c struct __arch_rwlock { unsigned int lock; }; typedef struct __arch_rwlock arch_rwlock_t; ``` 可以看到,在 x86_64 架构下,arch_rwlock_t 只包含一个 unsigned int 类型的 lock 成员变量,用于存储状态。 读写锁的初始化: ```c void rwlock_init(rwlock_t *lock) { raw_spin_lock_init(&lock->raw_lock.spinlock); atomic_long_set(&lock->raw_lock.rw_sem, 0); } ``` 其,raw_spin_lock_init() 用于初始化写锁,atomic_long_set() 用于初始化读计数器。 读的获取: ```c void read_lock(rwlock_t *lock) { while (1) { long count = atomic_long_read(&lock->raw_lock.rw_sem); if (count >= 0) { if (atomic_long_cmpxchg(&lock->raw_lock.rw_sem, count, count + 1) == count) { break; } } else { cpu_relax(); } } raw_spin_lock(&lock->raw_lock.spinlock); } ``` 其,atomic_long_read() 用于读取读计数器的值,如果值大于等于 0,则表示读可用,此时使用 atomic_long_cmpxchg() 原子操作来增加读计数器并获取读;如果值小于 0,则表示有写锁在使用,此时使用 cpu_relax() 函数等待写锁释放。 读的释放: ```c void read_unlock(rwlock_t *lock) { raw_spin_unlock(&lock->raw_lock.spinlock); atomic_long_dec(&lock->raw_lock.rw_sem); } ``` 其,raw_spin_unlock() 用于释放写锁,atomic_long_dec() 用于减少读计数器的值。 写锁的获取: ```c void write_lock(rwlock_t *lock) { raw_spin_lock(&lock->raw_lock.spinlock); while (1) { long count = atomic_long_read(&lock->raw_lock.rw_sem); if (count == 0) { if (atomic_long_cmpxchg(&lock->raw_lock.rw_sem, 0, -1) == 0) { break; } } else { cpu_relax(); } } } ``` 其,raw_spin_lock() 用于获取写锁,atomic_long_read() 用于读取读计数器的值,如果值等于 0,则表示读未被使用,此时使用 atomic_long_cmpxchg() 原子操作将读计数器的值修改为 -1 并获取写锁;如果值大于 0,则表示有读在使用,此时使用 cpu_relax() 函数等待读释放。 写锁的释放: ```c void write_unlock(rwlock_t *lock) { atomic_long_set(&lock->raw_lock.rw_sem, 0); raw_spin_unlock(&lock->raw_lock.spinlock); } ``` 其,atomic_long_set() 用于将读计数器的值设为 0,raw_spin_unlock() 用于释放写锁。 以上就是 Linux 内核读写锁的源码实现。值得注意的是,在多处理器环境下,读写锁的实现可能会涉及到更复杂的机制,例如优先等待、读者优先等待等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值