常用的简单的互斥的例子

总结一下, 我们写module的时候, 常用的简单的互斥的例子:

以下是个一个例子, 只有一个进程才能打开open() , 其他的进程不可以,直接返回。大家可以比较一下优劣。 
 


-------------------------------------------------
用信号量:

static DECLARE_MUTEX(char_sem);
static unsigned long char_inuse;

static int device_open(struct inode *inode, struct file *file)
{
    static int counter = 0;
    down_interruptible(&char_sem);

    if(char_inuse == 1) {
        up(&char_sem);
        return -EBUSY;
    }
    char_inuse = 1;

    up(&char_sem);
    
    ...........
}

static int device_release(struct inode *inode, struct file *file)
{
    char_inuse = 0;
    MOD_DEC_USE_COUNT;
    
    return 0;
}


-----------------------------------

用原子变量:
static atomic_t inuse_atomic = ATOMIC_INIT(-1);


/* 这个动作有可能并发,这里我又改成原子变量了
 */

static int device_open(struct inode *inode, struct file *file)
{
    
    if( ! atomic_inc_and_test(&inuse_atomic)) //原子加1 , 并判断是否==0 , -1 , 表示无人使用此设备

    {
        dbg("%d process is opening this device \n",inuse_atomic);//如果值是1 , 说明刚好有一个进程在访问该设备

        dbg("open device failure \n");
        return -EBUSY;
    }
    //try_module_get(THIS_MODULE);

    MOD_INC_USE_COUNT; 

    return SUCCESS;
}

static int device_release(struct inode *inode, struct file *file)
{

    atomic_set(&inuse_atomic,-1);//重新再把它置成-1

    MOD_DEC_USE_COUNT;
    
    return 0;
}

------------------------------
用原子的bit操作

static int chardev_users = 0; //防止互斥, 这里用更严格的test_and_set_bit() 函数


static int device_open(struct inode *inode, struct file *file)
{
    static int counter = 0;
    //    if (Device_Open)

    //        return -EBUSY;

    //    Device_Open++;

    //    上面的办法太不可靠了。用原子操作或者bit操作才是最可靠的。

    if(test_and_set_bit(0,&chardev_users))
        return -EBUSY;

    //sprintf(msg, "I already told you %d times Hello world!\n", counter++);

    //memset(msg,'\0',sizeof(msg));

    msg_Ptr = msg;
    //try_module_get(THIS_MODULE);

    MOD_INC_USE_COUNT; //increase the flag to present the module is used

/* strategy :
 * you should here allocate resources , not in init_module() ,if(refer == 0) 
 */
 

    return SUCCESS;
}

static int device_release(struct inode *inode, struct file *file)
{
    //    Device_Open--;        /* We're now ready for our next caller */


    //chardev_users =0; //直到现在才可以被别人open()

    //set_bit() is better by me

    if( !test_and_clear_bit(0,&chardev_users)) //bit0 should be 1 

        dbg("It looks wrong ,lock may bad \n");

    MOD_DEC_USE_COUNT;

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 互斥锁:对于一个写操作,可以使用互斥锁来阻止其他线程读取同一个变量或者资源。读写锁:如果多个线程同时访问一个变量或资源,可以使用读写锁来保证线程安全。条件变量:当某线程需要等待另一个线程完成某任务时,可以使用条件变量来实现。信号量:可以使用信号量来控制同时访问的线程数量,从而保证线程安全。屏障:当多个线程需要在某个时刻同步执行时,可以使用屏障来阻止其他线程继续执行,直到所有线程都到达屏障。 ### 回答2: 互斥锁、读写锁、条件变量、信号量、屏障是操作系统中常用的同步原语,用于多线程或多进程之间的协调和同步。下面举几个例子来说明它们的应用: 1. 互斥锁: 互斥锁是最常见且基础的同步原语,用来保护临界区,确保同一时间只有一个线程可以访问共享资源。例如,在多线程编程中,多个线程需要访问共享的全局变量,我们可以使用互斥锁来保证线程的互斥访问。 2. 读写锁: 读写锁是一种特殊的锁,它分为读锁和写锁。多个线程可以同时获取读锁,但只有一个线程可以获取写锁。读写锁适用于读多写少的场景,可以提高读操作的并发性。例如,在一个文件缓存系统中,多个线程可以同时读取缓存的文件内容,但只有一个线程可以写入缓存。 3. 条件变量: 条件变量用于线程间的等待和通知机制,能够在满足特定条件时唤醒等待的线程。例如,在生产者-消费者模型中,生产者需要等待缓冲区不满时才能继续生产,消费者需要等待缓冲区不空时才能继续消费。条件变量可以通过等待和通知的方式实现线程的同步。 4. 信号量: 信号量用于控制对临界资源的访问数量。可以将信号量看作是一个计数器,当资源被占用时,计数器减1,当资源被释放时,计数器加1。例如,在操作系统中,可以使用信号量来限制一个资源的并发访问数量,比如限制同时访问数据库的连接数量。 5. 屏障: 屏障用于控制多个线程在某个点上的同步,即在该点前的线程必须等待所有线程都到达该点才能继续执行。例如,在一个并行计算任务中,可能需要多个线程在某个阶段完成计算后再进入下一个阶段,这时可以使用屏障来同步各个线程的执行。 ### 回答3: 互斥锁是一种保护共享资源的锁,在同一时刻只允许一个线程对共享资源进行操作。例如,当多个线程同时访问一个共享计数器时,互斥锁可以保证每次只有一个线程能够增加或减少计数器的值,避免了竞态条件的发生。 读写锁是一种更高级别的锁,允许多个线程同时读取共享资源,但只允许一个线程进行写操作。例如,在一个并发读写文件的场景中,读写锁可以保证多个线程可以同时读取文件的内容,但只允许一个线程进行写入操作。 条件变量是一种用于线程间通信的机制,可以通知等待的线程某个特定的条件已经满足。例如,在一个生产者-消费者模型中,当缓冲区满时,生产者线程可以通过条件变量通知消费者线程可以消费数据了。 信号量是一种用于控制多个线程并发访问共享资源的机制。例如,当有限数量的资源需要在多个线程间共享时,可以使用信号量来限制资源的并发访问数量,从而避免资源的过度竞争。 屏障用于线程同步,可以让多个线程在某个特定的点上等待,直到所有线程都到达该点才能继续执行。例如,当多个线程并行执行任务,但需要等待所有线程都完成自己的任务后再进行下一步操作时,可以使用屏障来实现线程间的同步。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值