- 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
- 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
- 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
- 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们。
- 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
- 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
实验十一 内核同步实验
实验目的
掌握字符设备驱动程序编写的框架;
掌握内核中同步操作函数的使用;
实验原理
自旋锁:只有两种状态:锁上,没锁上。(lock, unlock)
当一个线程上锁自旋锁时,别的线程想要锁上该自旋锁的时候,会自旋(轮询)。直到第一个线程释放该自旋锁。
信号量:跟自旋锁相比:不能获得时,休眠(而不是自旋);
有数值(不是两种状态);
读写自旋锁:读可以多个线程一起;
但写操作时,只能一个线程;
读写信号量:休眠
互斥体:只有两种状态的信号量(1,0);
实验要求
(1)修改tiny4412开发板上的LED灯驱动程序(基于实验九);使用应用层打开设备文件时,同一时间,只能有一个进程能够使用该文件。一个进程打开设备文件后,另外一个进程打不开这个设备文件。
主要修改驱动的:open, release方法;
使用信号量实现;
可以让多个进程打开设备文件,但是读写操作不能同时进行,对驱动中要读写的全局变量进行同步保护;
主要修改驱动的:read,write方法;
分别使用:自旋锁,读写自旋锁,读写信号量,互斥体实现;
Leddrv:
#include linux/module.h
#include linux/kernel.h
#include linux/init.h
#include linux/fs.h
#include linux/types.h
#include linux/uaccess.h
#include linux/cdev.h
#include linux/device.h
#include linux/io.h
#include linux/atomic.h
#include linux/semaphore.h
#include linux/spinlock.h
#include linux/mutex.h
#include linux/rwlock.h
#include linux/rwsem.h
volatile unsigned long *VIR_GPM4CON;
volatile unsigned long *VIR_GPM4DAT;
static atomic_t led_atomic = ATOMIC_INIT(1);
DEFINE_SEMAPHORE(led_sema);
static int led_open(struct inode *in, struct file *fp)
{
/*
if(!atomic_dec_and_test(led_atomic))
{
atomic_inc(led_atomic);
return -EBUSY;
}*/
//down(led_sema);
down_interruptible(led_sema);
printk(kernel:led_open\n);
*VIR_GPM4CON = ~(0xffff);
*VIR_GPM4CON |= (0x1111);
*VIR_GPM4DAT = ~(0xf);
return 0;
}
static int led_release(struct inode *in, struct file *fp)
{
//atomic_inc(led_atomic);
up(led_sema);
printk(kernel:led_release\n);
*VIR_GPM4DAT |= (0xf);
return 0;
}
DEFINE_RWLOCK(led_rwlock);//定义读写自旋锁led_rwlock
static unsigned char led_state;
static ssize_t led_read(struct file *fp, char __user *buf, size_t len, loff_t * offset)
{
read_lock(led_rwlock);//上锁
led_state = led_state1;
copy_to_user(buf, led_state, 1);
read_unlock(led_rwlock);//解锁
return 1;
}
static ssize_t led_write(struct file *fp, const char __user *buf, size_t len, loff_t *offset)
{
write_lock(led_rwlock);//上锁
copy_from_user(led_state, buf, 1);
*VIR_GPM4DAT = ~(led_state);
write_unlock(led_rwlock);
文档评论(0)