linux驱动2.2按键中断-poll、异步通知、互斥、阻塞

一、poll机制

  为什么我们需要poll机制呢。之前的测试程序是这样:

while (1)
{
	read(fd, &key_val, 1);
	printf("key_val = 0x%x\n", key_val);
}

  在没有poll机制的情况下,大部分时间程序都处在read中休眠的那个位置。如果我们不想让程序停在这个位置,而是希望当有按键按下时,我们再去read,因此我们编写poll函数,测试程序调用poll函数根据返回值,来决定是否执行read函数。
  poll机制作用:相当于定时器,设置一定时间使进程等待资源,如果时间到了中断还处于睡眠状态(等待队列),poll机制就会唤醒中断,获取一次资源

1.1、poll机制内核框架

  如下图所示,在用户层上,使用poll或select函数时,和open、read那些函数一样,也要进入内核sys_poll函数里,接下来我们分析sys_poll函数来了解poll机制(位于/fs/select.c)

1)sys_poll代码如下:

asmlinkage long sys_poll(struct pollfd __user *ufds, unsigned int nfds,long timeout_msecs)
{
	if (timeout_msecs > 0)    //参数timeout>0
	{
		timeout_jiffies = msecs_to_jiffies(timeout_msecs);  //通过频率来计算timeout时间需要多少计数值
	}
	else
	{
		timeout_jiffies = timeout_msecs;    //如果timeout时间为0,直接赋值
	}
	return do_sys_poll(ufds, nfds, &timeout_jiffies);   //调用do_sys_poll。
}

2)然后进入do_sys_poll(位于fs/select.c):

int do_sys_poll(struct pollfd __user *ufds, unsigned int nfds, s64 *timeout)
{
  ...
  /*初始化一个poll_wqueues变量table*/
  poll_initwait(&table);
  ...
  fdcount = do_poll(nfds, head, &table, timeout);
  ...
}

3)进入poll_initwait函数,发现主要实现以下一句,后面会分析这里:

table ->pt-> qproc=__pollwait;    //__pollwait将在驱动的poll函数里的poll_wait函数用到

4)然后进入do_poll函数, (位于fs/select.c):

static int do_poll(unsigned int nfds,  struct poll_list *list, struct poll_wqueues *wait,  s64 *timeout)
{
...
	for (;;)
	{
...
		set_current_state(TASK_INTERRUPTIBLE);       //设置为等待队列状态
...
		for (; pfd != pfd_end; pfd++) {             //for循环运行多个poll机制
			/*将pfd和pt参数代入我们驱动程序里注册的poll函数*/
			if (do_pollfd(pfd, pt))     //若返回非0,count++,后面并退出
			{   count++;
				pt = NULL; } 
			}
...
			/*count非0(.poll函数返回非0),timeout超时计数到0,有信号在等待*/
			if (count || !*timeout || signal_pending(current))
				 break;
...
			/*进入休眠状态,只有当timeout超时计数到0,或者被中断唤醒才退出,*/
			__timeout = schedule_timeout(__timeout);
...
   }

	__set_current_state(TASK_RUNNING);  //开始运行
	return count;
}

4.1)上面do_pollfd函数到底是怎么将pfd和pt参数代入的?代码如下(位于fs/select.c):

static inline unsigned int do_pollfd(struct pollfd *pollfd, poll_table *pwait)
{
...
	if (file->f_op && file->f_op->poll)
	mask = file->f_op->poll(file, pwait);
...
	return mask;
}

  上面file->f_op 就是我们驱动里的file_oprations结构体,如下图所示:

  所以do_pollfd(pfd, pt)就执行了我们驱动程序里的.poll(pfd, pt)函数(第2小节开始分析.poll函数)
4.2)当poll进入休眠状态后,又是谁来唤醒它?这就要分析我们的驱动程序.poll函数(第1.2小节开始分析.poll函数)

1.2、编写并分析.poll函数

  在上一节驱动程序里添加以下代码:

#include <linux/poll.h>                //添加头文件
  
/*   .poll驱动函数:  third_poll        */
static unsigned int third_poll(struct file *fp, poll_table * wait)  //fp:文件  wait:
{
	unsigned int mask =0; 
	poll_wait(fp, &button_wait, wait);
	if(even_press)              //中断事件标志, 1:退出休眠状态     0:进入休眠状态 
		mask |= POLLIN | POLLRDNORM ;
	return mask;     //当超时,就返给应用层为0 ,被唤醒了就返回POLLIN | POLLRDNORM ;
}

static struct file_operations third_drv_fops={
	.owner = THIS_MODULE,
	.open = third_drv_open,
	.read = third_drv_read,
	.release=third_drv_class,   
	.poll = third_poll,           //创建.poll函数
};

1)在我们1.1-4)小节do_poll函数有一段以下代码:

if (do_pollfd(pfd, pt))     //若返回非0,count++,后面并退出
{      
	count++;
	pt = NULL;
}

  且在1.1-4.1)分析出: do_pollfd(pfd, pt)就是指向的驱动程序third_poll()函数,
  所以当我们有按键按下时, 驱动函数third_poll()就会返回mask非0值,然后在内核函数do_poll里的count就++,poll机制并退出睡眠.
2)分析在内核中poll机制如何被驱动里的中断唤醒的
  在驱动函数third_poll()里有以下一句:

poll_wait(fp, &button_wait, wait);


  如上图所示,代入参数,poll_wait()就是执行了: p->qproc(filp, button_wait, p);
刚好对应了我们1.1-3)小节的:

table ->pt-> qproc=__pollwait;

  所以poll_wait()函数就是调用了: __pollwait(filp, button_wait, p);
  然后我们来分析 __pollwait函数,pollwait的代码如下:

static void __pollwait(struct file  *filp, wait_queue_head_t  *wait_address,poll_table  *p)
{
   ... ...
   //把current进程挂载到&entry->wait下
   init_waitqueue_entry(&entry->wait, current);

   //再&entry->wait把添加到到button_wait中断下
   add_wait_queue(wait_address, &entry->wait);
}

  它是将poll进程添加到了button_wait中断队列里,这样,一有按键按下时,在中断服务函数里就会唤醒button_wait中断,同样也会唤醒poll机制,使poll机制重新进程休眠计数
3)驱动程序.poll函数返回值介绍
  当中断休眠状态时,返回mask为0
  当运行时返回:mask |= POLLIN | POLLRDNORM
  其中参数意义如下图:

  所以POLLIN | POLLRDNORM:普通数据可读|优先级带数据可读
  mask就返回到应用层poll函数,

1.3、改进测试程序third_poll_test.c(添加poll函数)

  在linux中可以通过man poll 来查看poll函数如何使用
  poll函数原型如下(#include <poll.h>):

int poll(struct pollfd *fds, nfds_t nfds, int timeout);

参数介绍:

  1. *fds:是一个poll描述符结构体数组(可以处理多个poll),结构体pollfd如下:
struct pollfd {
	int   fd;         /* file descriptor 文件描述符*/
	short events;     /* requested events 请求的事件*/
	short revents;    /* returned events 返回的事件(函数返回值)*/
};

  其中events和revents值参数如下图:

2) nfds:表示多少个poll,如果1个,就填入1
3) timeout:定时多少ms
  返回值介绍:
  返回值为0:表示超时或者fd文件描述符无法打开
  返回值为 -1:表示错误
  返回值为>0时 :就是以下几个常量

最终改进的测试代码如下:

#include <sys/types.h>    
#include <sys/stat.h>    
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <poll.h>                 //添加poll头文件

/*useg:    thirdtext   */
int main(int argc,char **argv)
{
	int fd,ret;
	unsigned int val=0;
	struct pollfd fds;                          //定义poll文件描述结构体               
	fd=open("/dev/buttons",O_RDWR);          

	if(fd<0)
	{	printf("can't open!!!\n");
		return -1;
	}

	fds.fd=fd;                   
	fds.events= POLLIN;           //请求类型是 普通或优先级带数据可读

	while(1)
	{
		ret=poll(&fds,1,5000) ;           //一个poll, 定时5000ms,进入休眠状态
		if(ret==0)                        //超时
		{
			printf("time out \r\n");
		}
		else if(ret>0)                   //poll机制被唤醒,表示有数据可读
		{
			read(fd,&val,1);         //读取一个值
			printf("key_val=0X%x\r\n",val);
		}  
	}
	return 0;
}

效果如下:

  如上图,若5S没有数据,则打印time out。

二、使用异步通知

之前学的应用层都是:

  • 1)查询方式:一直读
  • 2)中断方式.同样一直读,直到中断进程唤醒
  • 3)poll机制:一直在poll函数中睡眠,一定时间读一次

  以上3种,我们都是让应用程序主动去读,本节我们学习异步通知,它的作用就是当驱动层有数据时,主动告诉应用程序,然后应用程序再来读, 这样,应用程序就可以干其它的事情,不必一直读
  比如:kill -9 pid ,其实就是通过发信号杀死进程,kill发数据9给指定id号进程

2.1、怎么来收信号?

  通过signal函数来实现获取信号,先来看看以下例子:
头函数:

sighandler_t signal(int signum, sighandler_t handler); 

  函数说明:让一个信号与与一个函数对应,每当接收到这个信号就会调用相应的函数。
  头文件: #include <signal.h>
  参数1: 指明了所要处理的信号类型
  信号有以下几种:

  • SIGINT 键盘中断(如break、ctrl+c键被按下)
  • SIGUSR1 用户自定义信号1,kill的USR1(10)信号
  • SIGUSR2 用户自定义信号2, kill的USR2(12)信号

  参数2: 信号产生后需要处理的方式,可以是个函数
代码如下:

#include <stdio.h>
#include <signal.h>

void my_signal_run(int signum)       //信号处理函数
{
    static int run_cnt=0;
    printf("signal = %d, %d count\r\n",signum,++count);
}

int main(int argc,char **argv)
{
   signal(SIGUSR1,my_signal_run);  //调用signal函数,让指定的信号SIGUSR1与处理函数my_signal_run对应。
  while(1)
  {
		sleep(1000);     //去做其它事,睡眠1s
  }
   return 0;
}

  然后运行后,使用kill -10 802,可以看到产生单信号USR1(10)时就会调用my_signal_run()打印数据。

# kill -10 802
# signal = 10, 1 count

# kill -10 802
# signal = 10, 2 count

2.2、实现异步通知的要求

1)应用程序要实现有:注册信号处理函数,使用signal函数
2)谁来发?驱动来发
3)发给谁?驱动发给应用程序,但应用程序必须告诉驱动PID,
4)怎么发?驱动程序调用kill_fasync函数

2.3、写驱动函数

  我们在之前的中断程序上修改
1)定义“异步信号结构体”变量

static struct fasync_struct * button_async;

2)在file_operations结构体添加成员.fasync函数,并写函数

static struct file_operations third_drv_fops={
	.owner = THIS_MODULE,
	.open = fourth_drv_open,
	.read = fourth _drv_read,
	.release= fourth _drv_class,   
	.poll = fourth _poll,
	.fasync = fourth_fasync             //添加初始化异步信号函数
};

static int fourth_fasync (int fd, struct file *file, int on)
{
	return fasync_helper(fd, file, on, & button_async); //初始化button_async结构体,就能使用kill_fasync()了
}

  问:成员.fasync函数又是什么情况下使用?
  答:是被应用程序调用,在下面第2.4小节会见到。
3)在buttons_irq中断服务函数里发送信号:

kill_fasync(&button_async, SIGIO, POLL_IN);
//当有中断时,就发送SIGIO信号给应用层,应用层就会触发与SIGIO信号对应的函数

4)驱动程序代码如下:

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/irq.h>  
#include <asm/irq.h>
#include <asm/arch/regs-gpio.h>
#include <asm/hardware.h>
#include <asm/uaccess.h>
#include <asm/io.h>                      
#include <linux/poll.h>                  

static struct class *fourthdrv_class;                 
static struct class_device   *fourthdrv_class_devs; 

/*    声明等待队列类型中断 button_wait      */
static DECLARE_WAIT_QUEUE_HEAD(button_wait);

/*    异步信号结构体变量    */
static struct fasync_struct * button_async;

/*
* 定义中断事件标志
* 0:进入等待队列        1:退出等待队列
*/
static int even_press=0;                          
 
/*
* 定义全局变量key_val,保存key状态
*/
static int key_val=0;                          

/*
*引脚描述结构体
*/
struct pin_desc{
	unsigned int  pin;
	unsigned int  pin_status;
};

/*
*key初始状态(没有按下): 0x01,0x02,0x03,0x04
*key状态(按下):             0x81,0x82,0x83,0x84
*/

struct pin_desc  pins_desc[4]={
	{S3C2410_GPF0,0x01 },
	{S3C2410_GPF2, 0x02 },
	{S3C2410_GPG3, 0x03 },
	{S3C2410_GPG11,0x04},
} ;

int  fourth_drv_class(struct inode *inode, struct file  *file)  //卸载中断
{
    free_irq(IRQ_EINT0,&pins_desc[0]);
    free_irq(IRQ_EINT2,&pins_desc[1]);
    free_irq(IRQ_EINT11,&pins_desc[2]);
    free_irq(IRQ_EINT19,&pins_desc[3]);
    return 0;
}

/*
*   确定是上升沿还是下降沿
*/
static irqreturn_t  buttons_irq (int irq, void *dev_id)       //中断服务函数
{
    struct pin_desc *pindesc=(struct pin_desc *)dev_id;     //获取引脚描述结构体
    unsigned int  pin_val=0;                                              
    pin_val=s3c2410_gpio_getpin(pindesc->pin);
    if(pin_val)
	{
		/*没有按下 (下降沿),清除0x80*/        
		key_val=pindesc->pin_status&0xef;
	}
	else
	{
		/*按下(上升沿),加上0x80*/
		key_val=pindesc->pin_status|0x80;
	}
	even_press=1;                                        //退出等待队列
	wake_up_interruptible(&button_wait);                  //唤醒 中断
	kill_fasync(&button_async, SIGIO, POLL_IN);        //发送SIGIO信号给应用层
	return IRQ_RETVAL(IRQ_HANDLED);                   
}

static int fourth_drv_open(struct inode *inode, struct file  *file)
{
    request_irq(IRQ_EINT0,buttons_irq,IRQT_BOTHEDGE,"S1",&pins_desc[0]);
    request_irq(IRQ_EINT2, buttons_irq,IRQT_BOTHEDGE, "S2", &pins_desc[1]);
    request_irq(IRQ_EINT11, buttons_irq,IRQT_BOTHEDGE, "S3", &pins_desc[2]);
    request_irq(IRQ_EINT19, buttons_irq,IRQT_BOTHEDGE, "S4", &pins_desc[3]);
    return 0;
}

static int fourth_drv_read(struct file *file, const char __user *buf, size_t count, loff_t * ppos)
{       
/*将中断 进入等待队列(休眠状态)*/
	wait_event_interruptible(button_wait, even_press);           

	/*有按键按下,退出等待队列,上传key_val 给用户层*/
	if(copy_to_user(buf,&key_val,sizeof(key_val)))
		return EFAULT;

	even_press=0;      
	return 0;
}

static unsigned fourth_poll(struct file *file, poll_table *wait)
{
	unsigned int mask = 0;
	poll_wait(file, &button_wait, wait); // 不会立即休眠        
	if (even_press)
		mask |= POLLIN | POLLRDNORM;     

	return mask;
}

static int fourth_fasync (int fd, struct file *file, int on)
{
	return fasync_helper(fd, file, on, & button_async); //初始化button_async结构体,就能使用kill_fasync()了
}

static struct file_operations fourth_drv_fops={
	.owner = THIS_MODULE,
	.open = fourth_drv_open,
	.read = fourth_drv_read,
	.release=fourth_drv_class,    //里面添加free_irq函数,来释放中断服务函数
	.poll = fourth_poll,
	.fasync= fourth_fasync,     //初始化异步信号函数
};

volatile int fourth_major;
static int fourth_drv_init(void)
{
    fourth_major=register_chrdev(0,"fourth_drv",&fourth_drv_fops);  //创建驱动
    fourthdrv_class=class_create(THIS_MODULE,"fourth_dev");    //创建类名
    fourthdrv_class_devs=class_device_create(fourthdrv_class, NULL, MKDEV(fourth_major,0), NULL,"buttons");  
    return 0;
}

static int fourth_drv_exit(void)
{
	unregister_chrdev(fourth_major,"fourth_drv");            //卸载驱动
	class_device_unregister(fourthdrv_class_devs);         //卸载类设备
	class_destroy(fourthdrv_class);                              //卸载类
	return 0;
}

module_init(fourth_drv_init);
module_exit(fourth_drv_exit);
MODULE_LICENSE("GPL v2");

2.4、写应用测试程序

步骤如下:
1)signal(SIGIO, my_signal_fun);
  调用signal函数,当接收到SIGIO信号就进入my_signal_fun函数,读取驱动层的数据
2) fcntl(fd,F_SETOWN,getpid());
  指定进程做为fd文件的”属主”,内核收到F_SETOWN命令,就会设置pid(驱动无需处理),这样fd驱动程序就知道发给哪个进程
3) oflags=fcntl(fd,F_GETFL);
  获取fd的文件状态标志
4) fcntl(fd,F_SETFL, oflags| FASYNC );
  添加FASYNC状态标志,会调用驱动中成员.fasync函数,执行fasync_helper()来初始化异步信号结构体
  这4个步骤执行后,一旦有驱动层有SIGIO信号时,进程就会收到
应用层代码如下:

#include <sys/types.h>    
#include <sys/stat.h>    
#include <stdio.h>
#include <string.h>
#include <poll.h>               
#include <signal.h>
#include <unistd.h>
#include <fcntl.h>

int fd,ret;
void my_signal_fun(int signame)      //有信号来了
{
    read( fd, &ret, 1);              //读取驱动层数据
    printf("key_vale=0X%x\r\n",ret); 

}

/*useg:    fourthtext   */
int main(int argc,char **argv)
{
    int oflag;
    unsigned int val=0;      
    fd=open("/dev/buttons",O_RDWR); 
    if(fd<0)
	{
		printf("can't open!!!\n");
		return -1;
	}

    signal(SIGIO,my_signal_fun); //指定的信号SIGIO与处理函数my_signal_run对应
    fcntl( fd, F_SETOWN, getip());  //指定进程作为fd 的属主,发送pid给驱动
    oflag=fcntl( fd, F_GETFL);   //获取fd的文件标志状态
    fcntl( fd, F_SETFL, oflag|FASYNC);  //添加FASYNC状态标志,调用驱动层.fasync成员函数 

    while(1)
    {
		sleep(1000);                  //做其它的事情
    }
    return 0;
}

2.5、运行查看结果

三、互斥与阻塞机制

本节目标:
  1)学习原子操作和互斥信号量,实现互斥机制,同一时刻只能一个应用程序使用驱动程序
  2)学习阻塞和非阻塞操作
  当设备被一个程序打开时,存在被另一个程序打开的可能,如果两个或多个程序同时对设备文件进行写操作,这就是说我们的设备资源同时被多个进程使用,对共享资源(硬件资源、和软件上的全局变量、静态变量等)的访问则很容易导致竞态。
  显然这不是我们想要的,所以本节引入互斥的概念:实现同一时刻,只能一个应用程序使用驱动程序
  互斥其实现很简单,就是采用一些标志,当文件被一个进程打开后,就会设置该标志,使其他进程无法打开设备文件。

3.1、使用标志来实现互斥(不好)

  其中的标志需要使用函数来操作,不能直接通过判断变量来操作标志,比如:

if (-- canopen != 0)  //当canopen==0,表示没有进程访问驱动,当canopen<0:表示有进程访问

  编译汇编来看,分了3段: 读值、减1、判断
  如果刚好在读值的时候发生了中断,有另一个进程访问时,那么也会访问成功,也会容易导致访问竞态。
  所以采用某种函数来实现,保证执行过程不被其他行为打断,有两种类型函数可以实现:

  • 原子操作(像原子一样不可再细分不可被中途打断)
      当多个进程同时访问同一个驱动时,只能有一个进程访问成功,其它进程会退出
  • 互斥信号量操作
      比如:A、B进程同时访问同一个驱动时,只有A进程访问成功了,B进程进入休眠等待状态,当A进程执行完毕释放后,等待状态的B进程又来访问,保证一个一个进程都能访问

3.2、原子操作详解

  原子操作指的是在执行过程中不会被别的代码路径所中断的操作。
原子操作函数如下:

1)atomic_t v = ATOMIC_INIT(0);         //定义原子变量v并初始化为0
2)atomic_read(atomic_t *v);            //返回原子变量的值
3)void atomic_inc(atomic_t *v);        //原子变量增加1
4)void atomic_dec(atomic_t *v);        //原子变量减少1
5)int atomic_dec_and_test(atomic_t *v); //自减操作后测试其是否为0,为0则返回true,否则返回false。

3.2.1、修改驱动程序

1)定义原子变量:

/*定义原子变量canopen并初始化为1 */
atomic_t canopen = ATOMIC_INIT(1);  

2)在.open成员函数里添加:

/*自减操作后测试其是否为0,为0则返回true,否则返回false   */
if(!atomic_dec_and_test(&canopen))     
{
	atomic_inc(&canopen);       //++,复位
	return -1;
}

3)在. release成员函数里添加:

	atomic_inc(&canopen);       //++,复位

3.2.2、修改测试程序

int main(int argc,char **argv)
{
	int oflag;
	unsigned int val=0;     
	fd=open("/dev/buttons",O_RDWR);          
	if(fd<0)
	{
		printf("can't open, fd=%d\n",fd);
		return -1;
	}
	while(1)
	{ 
		read( fd, &ret, 1);              //读取驱动层数据
		printf("key_vale=0X%x\r\n",ret);   
	}  
	return 0;
}

3.2.3、测试效果

  如下图,可以看到第一个进程访问驱动成功,后面的就再也不能访问成功了

3.3、互斥信号量详解

  互斥信号量(semaphore)是用于保护临界区的一种常用方法,只有得到信号量的进程才能执行临界区代码。
  当获取不到信号量时,进程进入休眠等待状态。
信号量函数如下:


/*注意: 在2.6.36版本后这个函数DECLARE_MUTEX修改成DEFINE_SEMAPHORE了*/
1)static DECLARE_MUTEX(button_lock);     //定义互斥锁button_lock,被用来后面的down和up用
2)void down(struct semaphore * sem);            // 获取不到就进入不被中断的休眠状态(down函数中睡眠)
3)int down_interruptible(struct semaphore * sem);  //获取不到就进入可被中断的休眠状态(down函数中睡眠)
4)int down_trylock(struct semaphore * sem);       //试图获取信号量,获取不到则立刻返回正数
5)void up(struct semaphore * sem);               //释放信号量

3.3.1、修改驱动程序(以down函数获取为例)

1)定义互斥锁变量:

/*定义互斥锁button_lock,被用来后面的down()和up()使用 */
static DECLARE_MUTEX(button_lock);  

2)在.open成员函数里添加:

/* 获取不到就进入不被中断的休眠状态(down函数中睡眠) */
	down(&button_lock);

3)在. release成员函数里添加:

 /*         释放信号量          */
	up(&button_lock);

3.3.2、修改测试程序

int main(int argc,char **argv)
{
	int oflag;
	unsigned int val=0;      
	fd=open("/dev/buttons",O_RDWR);          
	if(fd<0)
	{
		printf("can't open, fd=%d\n",fd);
		return -1;
	}
	else
	{
		printf("can open,PID=%d\n",getpid());    //打开成功,打印pid进程号
	}
	while(1)
	{ 
		read( fd, &ret, 1);              //读取驱动层数据
		printf("key_vale=0X%x\r\n",ret);  
	}  
	return 0;
}

3.3.3、测试效果

  如下图所示,3个进程同时访问时,只有一个进程访问成功,其它2个进程进入休眠等待状态

  如下图所示,多个信号量访问时, 会一个一个进程来排序访问

3.4、阻塞与非阻塞

3.4.1、阻塞操作

  进程进行设备操作时,使用down()函数,若获取不到资源则挂起进程,将被挂起的进程进入休眠状态,被从调度器的运行队列移走,直到等待的条件被满足。
  在read读取按键时, 一直等待按键按下才返回数据

3.4.2、非阻塞操作

  进程进行设备操作时,使用down_trylock()函数,若获取不到资源并不挂起,直接放弃。
  在read读取按键时, 不管有没有数据都要返回

3.4.3、怎么来判断阻塞与非阻塞操作?

  在用户层open时,默认为阻塞操作,如果添加了” O_NONBLOCK”,表示使open()、read()、write()不被阻塞
实例:

fd=open("/dev/buttons",O_RDWR);                                          //使用阻塞操作
fd = open("/dev/buttons ", O_RDWR | O_NONBLOCK);                         //使用非阻塞操作

  然后在驱动设备中,通过file_operations成员函数.open、.read、.write带的参数file->f_flags 来查看用户层访问时带的参数
实例:

    if(  file->f_flags & O_NONBLOCK )   //非阻塞操作,获取不到则退出
    {
		... ...
	}
    else   //阻塞操作,获取不到则进入休眠
    {
		... ...
    }

3.4.4、修改应用程序,通过判断file->f_flags来使用阻塞操作还是非阻塞操作

1)定义互斥锁变量:

/*定义互斥锁button_lock,被用来后面的down()和up()使用 */
static DECLARE_MUTEX(button_lock); 

2)在.open成员函数里添加:

if( file->f_flags & O_NONBLOCK )   //非阻塞操作
{
	if(down_trylock(&button_lock) )       //尝试获取信号量,获取不到则退出
	return -1;
}
else   //阻塞操作
{
	down(&button_lock);         //获取信号量,获取不到则进入休眠
}

3)在. release成员函数里添加:

	/*释放信号量*/
	up(&button_lock);

3.4.5、写阻塞测试程序 fifth_blocktest.c

代码如下:

int main(int argc,char **argv)
{
	int oflag;
	unsigned int val=0;       
	fd=open("/dev/buttons",O_RDWR);           //使用阻塞操作
	if(fd<0)
	{
		printf("can't open, fd=%d\n",fd);      
		return -1;
	}
	else
	{
		printf("can open,PID=%d\n",getpid());    //打开成功,打印pid进程号
	} 

	while(1)
	{ 
		val=read( fd, &ret, 1);              //读取驱动层数据
		printf("key_vale=0X%x,retrun=%d\r\n",ret,val);  
	}
	return 0;
}

3.4.6、非阻塞测试效果

如下图所示:

3.4.7、写阻塞测试程序 fifth_nonblock.c

代码如下:

int main(int argc,char **argv)
{
	int oflag;
	unsigned int val=0;      
	fd=open("/dev/buttons",O_RDWR | O_NONBLOCK);   //使用非阻塞操作 
	if(fd<0)
	{
		printf("can't open, fd=%d\n",fd);
		return -1;
	}
	else
	{
		printf("can open,PID=%d\n",getpid());    //打开成功,打印pid进程号
	}

	while(1)
	{ 
		val=read( fd, &ret, 1);                              //读取驱动层数据
		printf("key_vale=0X%x,retrun=%d\r\n",ret,val); 
		sleep(3);         //延时3S
	}  
	return 0;
}

3.4.8、非阻塞测试效果

如下图所示:

posted @ 2019-09-25 16:53  princepeng  阅读(972)  评论(0编辑  收藏  举报