linux 信号量 自旋锁 测试 实验,Linux设备驱动并发控制详解 自旋锁,信号量

这个博客介绍了Linux内核驱动程序的实现,特别是关于全局变量的并发访问控制。驱动注册了一个字符设备,并定义了读写、打开和释放操作。通过自旋锁和信号量确保了对全局变量`global_var`的互斥访问。当多个进程尝试打开设备文件`/dev/globalvar`时,只有一个进程能成功,其他进程会因资源忙而返回错误。这展示了Linux内核级别的并发控制机制。
摘要由CSDN通过智能技术生成

#include MODULE_LICENSE("GPL");

#define MAJOR_NUM 254

static ssize_t globalvar_read(struct file *, char *, size_t, loff_t*);

static ssize_t globalvar_write(struct file *, const char *, size_t, loff_t*);

static int globalvar_open(struct inode *inode, struct file *filp);

static int globalvar_release(struct inode *inode, struct file *filp);

struct file_operations globalvar_fops =

{

read: globalvar_read, write: globalvar_write, open: globalvar_open, release:

globalvar_release,

};

static int global_var = 0;

static int globalvar_count = 0;

static struct semaphore sem;

static spinlock_t spin = SPIN_LOCK_UNLOCKED;

static int __init globalvar_init(void)

{

int ret;

ret = register_chrdev(MAJOR_NUM, "globalvar", &globalvar_fops);

if (ret)

{

printk("globalvar register failure");

}

else

{

printk("globalvar register success");

init_MUTEX(&sem);

}

return ret;

}

static void __exit globalvar_exit(void)

{

int ret;

ret = unregister_chrdev(MAJOR_NUM, "globalvar");

if (ret)

{

printk("globalvar unregister failure");

}

else

{

printk("globalvar unregister success");

}

}

static int globalvar_open(struct inode *inode, struct file *filp)

{

//获得自选锁

spin_lock(&spin);

//临界资源访问

if (globalvar_count)

{

spin_unlock(&spin);

return - EBUSY;

}

globalvar_count++;

//释放自选锁

spin_unlock(&spin);

return 0;

}

static int globalvar_release(struct inode *inode, struct file *filp)

{

globalvar_count--;

return 0;

}

static ssize_t globalvar_read(struct file *filp, char *buf, size_t len, loff_t

*off)

{

if (down_interruptible(&sem))

{

return - ERESTARTSYS;

}

if (copy_to_user(buf, &global_var, sizeof(int)))

{

up(&sem);

return - EFAULT;

}

up(&sem);

return sizeof(int);

}

static ssize_t globalvar_write(struct file *filp, const char *buf, size_t len,

loff_t *off)

{

if (down_interruptible(&sem))

{

return - ERESTARTSYS;

}

if (copy_from_user(&global_var, buf, sizeof(int)))

{

up(&sem);

return - EFAULT;

}

up(&sem);

return sizeof(int);

}

module_init(globalvar_init);

module_exit(globalvar_exit);

为了上述驱动程序的效果,我们启动两个进程分别打开/dev/globalvar。在两个终端中调用./globalvartest.o测试程序,当一个进程打开/dev/globalvar后,另外一个进程将打开失败,输出"device open failure",如下图:

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值