Linux 设备驱动中的 I/O模型(一)—— 阻塞和非阻塞I/O

在前面学习网络编程时,曾经学过I/O模型 Linux 系统应用编程——网络编程(I/O模型),下面学习一下I/O模型在设备驱动中的应用。

       回顾一下在Unix/Linux下共有五种I/O模型,分别是:

a -- 阻塞I/O
b -- 非阻塞I/O
c -- I/O复用(select和poll)
d -- 信号驱动I/O(SIGIO)
e -- 异步I/O(Posix.1的aio_系列函数)

 

       下面我们先学习阻塞I/O、非阻塞I/O 、I/O复用(select和poll),先学习一下基础概念

a -- 阻塞 

       阻塞操作是指在执行设备操作时,若不能获得资源,则挂起进程知道满足可操作的条件后再进行操作;被挂起的进程进入休眠状态(放弃CPU),被从调度器的运行队列移走,直到等待的条件被满足; 

b -- 非阻塞

      非阻塞的进程在不能进行设备操作时,并不挂起(继续占用CPU),它或者放弃,或者不停地查询,直到可以操作为止;

      二者的区别可以看应用程序的调用是否立即返回


      驱动程序通常需要提供这样的能力:当应用程序进行 read()、write() 等系统调用时,若设备的资源不能获取,而用户又希望以阻塞的方式访问设备,驱动程序应在设备驱动的xxx_read()、xxx_write() 等操作中将进程阻塞直到资源可以获取,此后,应用程序的 read()、write() 才返回,整个过程仍然进行了正确的设备 访问,用户并没感知到;若用户以非阻塞的方式访问设备文件,则当设备资源不可获取时,设备驱动的 xxx_read()、xxx_write() 等操作立刻返回, read()、write() 等系统调用也随即被返回

      因为阻塞的进程会进入休眠状态,因此,必须确保有一个地方能够唤醒休眠的进程,否则,进程就真的挂了。唤醒进程的地方最大可能发生在中断里面因为硬件资源获得的同时往往伴随着一个中断

      阻塞I/O通常由等待队列来实现,而非阻塞I/O由轮询来实现。



一、阻塞I/O实现 —— 等待队列

1、基础概念

       在Linux 驱动程序中,可以使用等待队列(wait queue)来实现阻塞进程的唤醒。wait queue 很早就作为一个基本的功能单位出现在Linux 内核里了,它以队列为基础数据结构,与进程调度机制紧密结合,能够实现内核中的异步事件通知机制。等待队列可以用来同步对系统资源的访问,上一篇文章所述的信号量在内核中也依赖等待队列来实现。

      在Linux内核中使用等待队列的过程很简单,首先定义一个wait_queue_head,然后如果一个task想等待某种事件,那么调用wait_event(等待队列,事件)就可以了。

      等待队列应用广泛,但是内核实现却十分简单。其涉及到两个比较重要的数据结构:__wait_queue_head,该结构描述了等待队列的链头,其包含一个链表和一个原子锁,结构定义如下:    

struct __wait_queue_head 
{
	 spinlock_t lock;                    /* 保护等待队列的原子锁 */
	 struct list_head task_list;         /* 等待队列 */
};
typedef struct __wait_queue_head wait_queue_head_t;

__wait_queue,该结构是对一个等待任务的抽象。每个等待任务都会抽象成一个wait_queue,并且挂载到wait_queue_head上。该结构定义如下:

struct __wait_queue 
{
	unsigned int flags;
	void *private;                       /* 通常指向当前任务控制块 */	/* 任务唤醒操作方法,该方法在内核中提供,通常为autoremove_wake_function */
	wait_queue_func_t func;             
	struct list_head task_list;              /* 挂入wait_queue_head的挂载点 */
};


    Linux中等待队列的实现思想如下图所示,当一个任务需要在某个wait_queue_head上睡眠时,将自己的进程控制块信息封装到wait_queue中,然后挂载到wait_queue的链表中,执行调度睡眠。当某些事件发生后,另一个任务(进程)会唤醒wait_queue_head上的某个或者所有任务,唤醒工作也就是将等待队列中的任务设置为可调度的状态,并且从队列中删除。




       使用等待队列时首先需要定义一个wait_queue_head,这可以通过DECLARE_WAIT_QUEUE_HEAD宏来完成,这是静态定义的方法。该宏会定义一个wait_queue_head,并且初始化结构中的锁以及等待队列。当然,动态初始化的方法也很简单,初始化一下锁及队列就可以了。

       一个任务需要等待某一事件的发生时,通常调用wait_event,该函数会定义一个wait_queue,描述等待任务,并且用当前的进程描述块初始化wait_queue,然后将wait_queue加入到wait_queue_head中。

函数实现流程说明如下:

a -- 用当前的进程描述块(PCB)初始化一个wait_queue描述的等待任务。

b -- 在等待队列锁资源的保护下,将等待任务加入等待队列。

c -- 判断等待条件是否满足,如果满足,那么将等待任务从队列中移出,退出函数。

d --  如果条件不满足,那么任务调度,将CPU资源交与其它任务。

e -- 当睡眠任务被唤醒之后,需要重复b、c 步骤,如果确认条件满足,退出等待事件函数。


2、等待队列接口函数

1、定义并初始化

/* 定义“等待队列头” */
wait_queue_head_t my_queue;
/* 初始化“等待队列头”*/
init_waitqueue_head(&my_queue);

直接定义并初始化。init_waitqueue_head()函数会将自旋锁初始化为未锁,等待队列初始化为空的双向循环链表。

DECLARE_WAIT_QUEUE_HEAD(my_queue); 定义并初始化,可以作为定义并初始化等待队列头的快捷方式。


2、定义等待队列:

DECLARE_WAITQUEUE(name,tsk);

定义并初始化一个名为name的等待队列。


3、(从等待队列头中)添加/移出等待队列:

/* add_wait_queue()函数,设置等待的进程为非互斥进程,并将其添加进等待队列头(q)的队头中*/
void add_wait_queue(wait_queue_head_t *q, wait_queue_t *wait);
/* 该函数也和add_wait_queue()函数功能基本一样,只不过它是将等待的进程(wait)设置为互斥进程。*/
void add_wait_queue_exclusive(wait_queue_head_t *q, wait_queue_t *wait);


4、等待事件:

(1)wait_event()宏:

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. /** 
  2.  * wait_event - sleep until a condition gets true 
  3.  * @wq: the waitqueue to wait on 
  4.  * @condition: a C expression for the event to wait for 
  5.  * 
  6.  * The process is put to sleep (TASK_UNINTERRUPTIBLE) until the 
  7.  * @condition evaluates to true. The @condition is checked each time 
  8.  * the waitqueue @wq is woken up. 
  9.  * 
  10.  * wake_up() has to be called after changing any variable that could 
  11.  * change the result of the wait condition. 
  12.  */  
  13.   
  14. #define wait_event(wq, condition)                   \  
  15.   
  16. do {                                    \  
  17.     if (condition)                          \  
  18.         break;                          \  
  19.      __wait_event(wq, condition);                    \  
  20. while (0)  

    在等待会列中睡眠直到condition为真。在等待的期间,进程会被置为TASK_UNINTERRUPTIBLE进入睡眠,直到condition变量变为真。每次进程被唤醒的时候都会检查condition的值.

(2)wait_event_interruptible()函数:

   和wait_event()的区别是调用该宏在等待的过程中当前进程会被设置为TASK_INTERRUPTIBLE状态.在每次被唤醒的时候,首先检查condition是否为真,如果为真则返回,否则检查如果进程是被信号唤醒,会返回-ERESTARTSYS错误码.如果是condition为真,则返回0.

(3)wait_event_timeout()宏:

   也与wait_event()类似.不过如果所给的睡眠时间为负数则立即返回.如果在睡眠期间被唤醒,且condition为真则返回剩余的睡眠时间,否则继续睡眠直到到达或超过给定的睡眠时间,然后返回0
(4)wait_event_interruptible_timeout()宏:
   与wait_event_timeout()类似,不过如果在睡眠期间被信号打断则返回ERESTARTSYS错误码.
(5) wait_event_interruptible_exclusive()宏
   同样和wait_event_interruptible()一样,不过该睡眠的进程是一个互斥进程.


5、唤醒队列

(1)wake_up()函数

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. #define wake_up(x)          __wake_up(x, TASK_NORMAL, 1, NULL)  
  2. /** 
  3.  * __wake_up - wake up threads blocked on a waitqueue. 
  4.  * @q: the waitqueue 
  5.  * @mode: which threads 
  6.  * @nr_exclusive: how many wake-one or wake-many threads to wake up 
  7.  * @key: is directly passed to the wakeup function 
  8.  */  
  9. void __wake_up(wait_queue_head_t *q, unsigned int mode,  
  10.             int nr_exclusive, void *key)  
  11. {  
  12.     unsigned long flags;  
  13.    
  14.     spin_lock_irqsave(&q->lock, flags);  
  15.     __wake_up_common(q, mode, nr_exclusive, 0, key);  
  16.     spin_unlock_irqrestore(&q->lock, flags);  
  17. }  
  18. EXPORT_SYMBOL(__wake_up);  
唤醒等待队列.可唤醒处于TASK_INTERRUPTIBLE和TASK_UNINTERUPTIBLE状态的进程,和wait_event/wait_event_timeout成对使用. (2)wake_up_interruptible()函数:

#define wake_up_interruptible(x)    __wake_up(x, TASK_INTERRUPTIBLE, 1, NULL)

和wake_up()唯一的区别是它只能唤醒TASK_INTERRUPTIBLE状态的进程.,与wait_event_interruptible/wait_event_interruptible_timeout/ wait_event_interruptible_exclusive成对使用。

下面看一个实例:

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. static ssize_t hello_read(struct file *filep, char __user *buf, size_t len, loff_t *pos)  
  2. {  
  3. /* 
  4.     实现应用进程read的时候,如果没有数据就阻塞 
  5. */  
  6.     if(len>64)  
  7.     {  
  8.         len =64;  
  9.     }  
  10.     wait_event_interruptible(wq, have_data == 1);  
  11.   
  12.     if(copy_to_user(buf,temp,len))  
  13.     {  
  14.         return -EFAULT;  
  15.     }      
  16.     have_data = 0;  
  17.     return len;  
  18. }  
  19. static ssize_t hello_write(struct file *filep, const char __user *buf, size_t len, loff_t *pos)  
  20. {  
  21.     if(len > 64)  
  22.     {  
  23.         len = 64;  
  24.     }  
  25.   
  26.     if(copy_from_user(temp,buf,len))  
  27.     {  
  28.         return -EFAULT;  
  29.     }  
  30.     printk("write %s\n",temp);  
  31.     have_data = 1;  
  32.     wake_up_interruptible(&wq);  
  33.     return len;  
  34. }  

注意两个概念:

a --  疯狂兽群

      wake_up的时候,所有阻塞在队列的进程都会被唤醒,但是因为condition的限制,只有一个进程得到资源,其他进程又会再次休眠,如果数量很大,称为 疯狂兽群

b -- 独占等待

      等待队列的入口设置一个WQ_FLAG_EXCLUSIVE标志,就会添加到等待队列的尾部,没有设置设置的添加到头部,wake up的时候遇到第一个具有WQ_FLAG_EXCLUSIVE这个标志的进程就停止唤醒其他进程



二、非阻塞I/O实现方式 —— 多路复用

1、轮询的概念和作用

      在用户程序中,select() 和 poll() 也是设备阻塞和非阻塞访问息息相关的论题。使用非阻塞I/O的应用程序通常会使用select() 和 poll() 系统调用查询是否可对设备进行无阻塞的访问。select() 和 poll() 系统调用最终会引发设备驱动中的 poll()函数被执行。


2、应用程序中的轮询编程

      在用户程序中,select()和poll()本质上是一样的, 不同只是引入的方式不同,前者是在BSD UNIX中引入的,后者是在System V中引入的。用的比较广泛的是select系统调用。原型如下

int select(int numfds, fd_set *readfds, fd_set *writefds, fd_set *exceptionfds, struct timeval *timeout);

    其中readfs,writefds,exceptfds分别是select()监视的读,写和异常处理的文件描述符集合,numfds的值是需要检查的号码最高的文件描述符加1,timeout则是一个时间上限值,超过该值后,即使仍没有描述符准备好也会返回。

struct timeval
{int tv_sec;    //秒int tv_usec;   //微秒
}

涉及到文件描述符集合的操作主要有以下几种:

1)清除一个文件描述符集   FD_ZERO(fd_set *set);

2)将一个文件描述符加入文件描述符集中    FD_SET(int fd,fd_set *set);

3)将一个文件描述符从文件描述符集中清除  FD_CLR(int fd,fd_set *set);

4)判断文件描述符是否被置位    FD_ISSET(int fd,fd_set *set);

最后我们利用上面的文件描述符集的相关来写个验证添加了设备轮询的驱动,把上边两块联系起来


3、设备驱动中的轮询编程

       设备驱动中的poll() 函数原型如下

unsigned int(*poll)(struct file *filp, struct poll_table * wait);

第一个参数是file结构体指针,第二个参数是轮询表指针,poll设备方法完成两件事:

a -- 对可能引起设备文件状态变化的等待队列调用poll_wait()函数,将对应的等待队列头添加到poll_table,如果没有文件描述符可用来执行 I/O, 则内核使进程在传递到该系统调用的所有文件描述符对应的等待队列上等待

b -- 返回表示是否能对设备进行无阻塞读、写访问的掩码。

位掩码:POLLRDNORM, POLLIN,POLLOUT,POLLWRNORM

设备可读,通常返回:(POLLIN | POLLRDNORM)

设备可写,通常返回:(POLLOUT | POLLWRNORM)

       

poll_wait()函数:用于向 poll_table注册等待队列

 void poll_wait(struct file *filp, wait_queue_head_t *queue,poll_table *wait)  

      poll_wait()函数不会引起阻塞,它所做的工作是把当前进程添加到wait 参数指定的等待列表(poll_table)中。

     真正的阻塞动作是上层的select/poll函数中完成的。select/poll会在一个循环中对每个需要监听的设备调用它们自己的poll支持函数以使得当前进程被加入各个设备的等待列表。若当前没有任何被监听的设备就绪,则内核进行调度(调用schedule)让出cpu进入阻塞状态,schedule返回时将再次循环检测是否有操作可以进行,如此反复;否则,若有任意一个设备就绪,select/poll都立即返回。

具体过程如下:

a -- 用户程序第一次调用select或者poll,驱动调用poll_wait并使两条队列都加入poll_table结构中作为下次调用驱动函数poll的条件,一个mask返回值指示设备是否可操作,0为未准备状态,如果文件描述符未准备好可读或可写,用户进程被会加入到写或读等待队列中进入睡眠状态

b -- 当驱动执行了某些操作,例如,写缓冲或读缓冲,写缓冲使读队列被唤醒,读缓冲使写队列被唤醒,于是select或者poll系统调用在将要返回给用户进程时再次调用驱动函数poll,驱动依然调用poll_wait 并使两条队列都加入poll_table结构中,并判断可写或可读条件是否满足,如果mask返回POLLIN | POLLRDNORM或POLLOUT | POLLWRNORM则指示可读或可写,这时select或poll真正返回给用户进程,如果mask还是返回0,则系统调用select或poll继续不返回

     

下面是一个典型模板:

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. static unsigned int XXX_poll(struct file *filp, poll_table *wait)  
  2. {  
  3.     unsigned int mask = 0;  
  4.         struct XXX_dev *dev = filp->private_data;     //获得设备结构指针  
  5.     ...  
  6.     poll_wait(filp, &dev->r_wait, wait);    //加读等待对列头  
  7.     poll_wait(filp ,&dev->w_wait, wait);    //加写等待队列头  
  8.       
  9.     if(...)//可读  
  10.     {  
  11.           mask |= POLLIN | POLLRDNORM;    //标识数据可获得  
  12.      }  
  13.     if(...)//可写  
  14.     {  
  15.           mask |= POLLOUT | POLLWRNORM;    //标识数据可写入  
  16.      }  
  17.     ..  
  18.     return mask;  
  19. }  

4、调用过程:

Linux下select调用的过程:

1、用户层应用程序调用select(),底层调用poll())
2、核心层调用sys_select() ------> do_select()

  最终调用文件描述符fd对应的struct file类型变量的struct file_operations *f_op的poll函数。
  poll指向的函数返回当前可否读写的信息。
  1)如果当前可读写,返回读写信息。
  2)如果当前不可读写,则阻塞进程,并等待驱动程序唤醒,重新调用poll函数,或超时返回。

3、驱动需要实现poll函数
当驱动发现有数据可以读写时,通知核心层,核心层重新调用poll指向的函数查询信息。

poll_wait(filp,&wait_q,wait) // 此处将当前进程加入到等待队列中,但并不阻塞
在中断中使用wake_up_interruptible(&wait_q)唤醒等待队列。

4、实例分析

1、memdev.h

/*mem设备描述结构体*/
struct mem_dev                                     
{                                                        char *data;                      unsigned long size; wait_queue_head_t inq;  
};#endif /* _MEMDEV_H_ */
2、驱动程序 memdev.c
[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. #include <linux/module.h>  
  2. #include <linux/types.h>  
  3. #include <linux/fs.h>  
  4. #include <linux/errno.h>  
  5. #include <linux/mm.h>  
  6. #include <linux/sched.h>  
  7. #include <linux/init.h>  
  8. #include <linux/cdev.h>  
  9. #include <asm/io.h>  
  10. #include <asm/system.h>  
  11. #include <asm/uaccess.h>  
  12.   
  13. #include <linux/poll.h>  
  14. #include "memdev.h"  
  15.   
  16. static mem_major = MEMDEV_MAJOR;  
  17. bool have_data = false/*表明设备有足够数据可供读*/  
  18.   
  19. module_param(mem_major, int, S_IRUGO);  
  20.   
  21. struct mem_dev *mem_devp; /*设备结构体指针*/  
  22.   
  23. struct cdev cdev;   
  24.   
  25. /*文件打开函数*/  
  26. int mem_open(struct inode *inode, struct file *filp)  
  27. {  
  28.     struct mem_dev *dev;  
  29.       
  30.     /*获取次设备号*/  
  31.     int num = MINOR(inode->i_rdev);  
  32.   
  33.     if (num >= MEMDEV_NR_DEVS)   
  34.             return -ENODEV;  
  35.     dev = &mem_devp[num];  
  36.       
  37.     /*将设备描述结构指针赋值给文件私有数据指针*/  
  38.     filp->private_data = dev;  
  39.       
  40.     return 0;   
  41. }  
  42.   
  43. /*文件释放函数*/  
  44. int mem_release(struct inode *inode, struct file *filp)  
  45. {  
  46.   return 0;  
  47. }  
  48.   
  49. /*读函数*/  
  50. static ssize_t mem_read(struct file *filp, char __user *buf, size_t size, loff_t *ppos)  
  51. {  
  52.   unsigned long p =  *ppos;  
  53.   unsigned int count = size;  
  54.   int ret = 0;  
  55.   struct mem_dev *dev = filp->private_data; /*获得设备结构体指针*/  
  56.   
  57.   /*判断读位置是否有效*/  
  58.   if (p >= MEMDEV_SIZE)  
  59.     return 0;  
  60.   if (count > MEMDEV_SIZE - p)  
  61.     count = MEMDEV_SIZE - p;  
  62.       
  63.   while (!have_data) /* 没有数据可读,考虑为什么不用if,而用while */  
  64.   {  
  65.         if (filp->f_flags & O_NONBLOCK)  
  66.             return -EAGAIN;  
  67.       
  68.     wait_event_interruptible(dev->inq,have_data);  
  69.   }  
  70.   
  71.   /*读数据到用户空间*/  
  72.   if (copy_to_user(buf, (void*)(dev->data + p), count))  
  73.   {  
  74.     ret =  - EFAULT;  
  75.   }  
  76.   else  
  77.   {  
  78.     *ppos += count;  
  79.     ret = count;  
  80.      
  81.     printk(KERN_INFO "read %d bytes(s) from %d\n", count, p);  
  82.   }  
  83.     
  84.   have_data = false/* 表明不再有数据可读 */  
  85.   /* 唤醒写进程 */  
  86.   return ret;  
  87. }  
  88.   
  89. /*写函数*/  
  90. static ssize_t mem_write(struct file *filp, const char __user *buf, size_t size, loff_t *ppos)  
  91. {  
  92.   unsigned long p =  *ppos;  
  93.   unsigned int count = size;  
  94.   int ret = 0;  
  95.   struct mem_dev *dev = filp->private_data; /*获得设备结构体指针*/  
  96.     
  97.   /*分析和获取有效的写长度*/  
  98.   if (p >= MEMDEV_SIZE)  
  99.     return 0;  
  100.   if (count > MEMDEV_SIZE - p)  
  101.     count = MEMDEV_SIZE - p;  
  102.   
  103.   /*从用户空间写入数据*/  
  104.   if (copy_from_user(dev->data + p, buf, count))  
  105.     ret =  - EFAULT;  
  106.   else  
  107.   {  
  108.     *ppos += count;  
  109.     ret = count;  
  110.       
  111.     printk(KERN_INFO "written %d bytes(s) from %d\n", count, p);  
  112.   }  
  113.     
  114.   have_data = true/* 有新的数据可读 */  
  115.       
  116.     /* 唤醒读进程 */  
  117.     wake_up(&(dev->inq));  
  118.   
  119.   return ret;  
  120. }  
  121.   
  122. /* seek文件定位函数 */  
  123. static loff_t mem_llseek(struct file *filp, loff_t offset, int whence)  
  124. {   
  125.     loff_t newpos;  
  126.   
  127.     switch(whence) {  
  128.       case 0: /* SEEK_SET */  
  129.         newpos = offset;  
  130.         break;  
  131.   
  132.       case 1: /* SEEK_CUR */  
  133.         newpos = filp->f_pos + offset;  
  134.         break;  
  135.   
  136.       case 2: /* SEEK_END */  
  137.         newpos = MEMDEV_SIZE -1 + offset;  
  138.         break;  
  139.   
  140.       default/* can't happen */  
  141.         return -EINVAL;  
  142.     }  
  143.     if ((newpos<0) || (newpos>MEMDEV_SIZE))  
  144.         return -EINVAL;  
  145.           
  146.     filp->f_pos = newpos;  
  147.     return newpos;  
  148.   
  149. }  
  150. unsigned int mem_poll(struct file *filp, poll_table *wait)  
  151. {  
  152.     struct mem_dev  *dev = filp->private_data;   
  153.     unsigned int mask = 0;  
  154.       
  155.    /*将等待队列添加到poll_table */  
  156.     poll_wait(filp, &dev->inq,  wait);  
  157.    
  158.       
  159.     if (have_data)         mask |= POLLIN | POLLRDNORM;  /* readable */  
  160.   
  161.     return mask;  
  162. }  
  163.   
  164.   
  165. /*文件操作结构体*/  
  166. static const struct file_operations mem_fops =  
  167. {  
  168.   .owner = THIS_MODULE,  
  169.   .llseek = mem_llseek,  
  170.   .read = mem_read,  
  171.   .write = mem_write,  
  172.   .open = mem_open,  
  173.   .release = mem_release,  
  174.   .poll = mem_poll,  
  175. };  
  176.   
  177. /*设备驱动模块加载函数*/  
  178. static int memdev_init(void)  
  179. {  
  180.   int result;  
  181.   int i;  
  182.   
  183.   dev_t devno = MKDEV(mem_major, 0);  
  184.   
  185.   /* 静态申请设备号*/  
  186.   if (mem_major)  
  187.     result = register_chrdev_region(devno, 2, "memdev");  
  188.   else  /* 动态分配设备号 */  
  189.   {  
  190.     result = alloc_chrdev_region(&devno, 0, 2, "memdev");  
  191.     mem_major = MAJOR(devno);  
  192.   }    
  193.     
  194.   if (result < 0)  
  195.     return result;  
  196.   
  197.   /*初始化cdev结构*/  
  198.   cdev_init(&cdev, &mem_fops);  
  199.   cdev.owner = THIS_MODULE;  
  200.   cdev.ops = &mem_fops;  
  201.     
  202.   /* 注册字符设备 */  
  203.   cdev_add(&cdev, MKDEV(mem_major, 0), MEMDEV_NR_DEVS);  
  204.      
  205.   /* 为设备描述结构分配内存*/  
  206.   mem_devp = kmalloc(MEMDEV_NR_DEVS * sizeof(struct mem_dev), GFP_KERNEL);  
  207.   if (!mem_devp)    /*申请失败*/  
  208.   {  
  209.     result =  - ENOMEM;  
  210.     goto fail_malloc;  
  211.   }  
  212.   memset(mem_devp, 0, sizeof(struct mem_dev));  
  213.     
  214.   /*为设备分配内存*/  
  215.   for (i=0; i < MEMDEV_NR_DEVS; i++)   
  216.   {  
  217.         mem_devp[i].size = MEMDEV_SIZE;  
  218.         mem_devp[i].data = kmalloc(MEMDEV_SIZE, GFP_KERNEL);  
  219.         memset(mem_devp[i].data, 0, MEMDEV_SIZE);  
  220.     
  221.       /*初始化等待队列*/  
  222.      init_waitqueue_head(&(mem_devp[i].inq));  
  223.      //init_waitqueue_head(&(mem_devp[i].outq));  
  224.   }  
  225.      
  226.   return 0;  
  227.   
  228.   fail_malloc:   
  229.   unregister_chrdev_region(devno, 1);  
  230.     
  231.   return result;  
  232. }  
  233.   
  234. /*模块卸载函数*/  
  235. static void memdev_exit(void)  
  236. {  
  237.   cdev_del(&cdev);   /*注销设备*/  
  238.   kfree(mem_devp);     /*释放设备结构体内存*/  
  239.   unregister_chrdev_region(MKDEV(mem_major, 0), 2); /*释放设备号*/  
  240. }  
  241.   
  242. MODULE_AUTHOR("David Xie");  
  243. MODULE_LICENSE("GPL");  
  244.   
  245. module_init(memdev_init);  
  246. module_exit(memdev_exit);  
3、应用程序 app-write.c
[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. #include <stdio.h>  
  2.   
  3. int main()  
  4. {  
  5.     FILE *fp = NULL;  
  6.     char Buf[128];  
  7.       
  8.       
  9.     /*打开设备文件*/  
  10.     fp = fopen("/dev/memdev0","r+");  
  11.     if (fp == NULL)  
  12.     {  
  13.         printf("Open Dev memdev Error!\n");  
  14.         return -1;  
  15.     }  
  16.       
  17.     /*写入设备*/  
  18.     strcpy(Buf,"memdev is char dev!");  
  19.     printf("Write BUF: %s\n",Buf);  
  20.     fwrite(Buf, sizeof(Buf), 1, fp);  
  21.       
  22.     sleep(5);  
  23.     fclose(fp);  
  24.       
  25.     return 0;      
  26.   
  27. }  

4、应用程序 app-read.c

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. #include <stdio.h>  
  2. #include <stdlib.h>  
  3. #include <unistd.h>  
  4. #include <sys/ioctl.h>  
  5. #include <sys/types.h>  
  6. #include <sys/stat.h>  
  7. #include <fcntl.h>  
  8. #include <sys/select.h>  
  9. #include <sys/time.h>  
  10. #include <errno.h>  
  11.   
  12. int main()  
  13. {  
  14.     int fd;  
  15.     fd_set rds;  
  16.     int ret;  
  17.     char Buf[128];  
  18.       
  19.     /*初始化Buf*/  
  20.     strcpy(Buf,"memdev is char dev!");  
  21.     printf("BUF: %s\n",Buf);  
  22.       
  23.     /*打开设备文件*/  
  24.     fd = open("/dev/memdev0",O_RDWR);  
  25.       
  26.     FD_ZERO(&rds);  
  27.     FD_SET(fd, &rds);  
  28.   
  29.     /*清除Buf*/  
  30.     strcpy(Buf,"Buf is NULL!");  
  31.     printf("Read BUF1: %s\n",Buf);  
  32.   
  33.     ret = select(fd + 1, &rds, NULL, NULL, NULL);  
  34.     if (ret < 0)   
  35.     {  
  36.         printf("select error!\n");  
  37.         exit(1);  
  38.     }  
  39.     if (FD_ISSET(fd, &rds))   
  40.         read(fd, Buf, sizeof(Buf));              
  41.       
  42.     /*检测结果*/  
  43.     printf("Read BUF2: %s\n",Buf);  
  44.       
  45.     close(fd);  
  46.       
  47.     return 0;      

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/402214.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Python 爬虫进阶六之多进程的用法

python 中的多线程其实并不是真正的多线程&#xff0c;并不能做到充分利用多核 CPU 资源。 如果想要充分利用&#xff0c;在 python 中大部分情况需要使用多进程&#xff0c;那么这个包就叫做 multiprocessing。 借助它&#xff0c;可以轻松完成从单进程到并发执行的转换。mult…

Linux 设备驱动的并发控制

Linux 设备驱动中必须要解决的一个问题是多个进程对共享的资源的并发访问&#xff0c;并发的访问会导致竞态&#xff0c;即使是经验丰富的驱动工程师也常常设计出包含并发问题bug 的驱动程序。 一、基础概念 1、Linux 并发相关基础概念 a -- 并发&#xff08;concurrency&#…

第三章:多坐标系

第一节&#xff1a;为什么要有多坐标系 当我们使用一个坐标系来描绘整个场景的时候&#xff0c;场景中的任意点都可以用该坐标系描述&#xff0c;此时如果有一只羊一遍摇动着耳朵&#xff0c;一边走&#xff0c;这个时候如果进行坐标的转换会发现异常的麻烦&#xff0c;此时如果…

Linux 设备驱动开发 —— 设备树在platform设备驱动中的使用

关与设备树的概念&#xff0c;我们在Exynos4412 内核移植&#xff08;六&#xff09;—— 设备树解析 里面已经学习过&#xff0c;下面看一下设备树在设备驱动开发中起到的作用 Device Tree是一种描述硬件的数据结构&#xff0c;设备树源(Device Tree Source)文件&#xff08;以…

Android 网络通信框架Volley简介(Google IO 2013)

1. 什么是Volley 在这之前&#xff0c;我们在程序中需要和网络通信的时候&#xff0c;大体使用的东西莫过于AsyncTaskLoader&#xff0c;HttpURLConnection&#xff0c;AsyncTask&#xff0c;HTTPClient&#xff08;Apache&#xff09;等&#xff0c;今年的Google I/O 2013上&…

Linux 设备驱动开发 —— platform设备驱动应用实例解析

前面我们已经学习了platform设备的理论知识Linux 设备驱动开发 —— platform 设备驱动 &#xff0c;下面将通过一个实例来深入我们的学习。 一、platform 驱动的工作过程 platform模型驱动编程&#xff0c;需要实现platform_device(设备)与platform_driver&#xff08;驱动&am…

Python爬虫入门四urllib库的高级用法

1.设置headers 有些网站不会同意程序直接用上面的方式进行访问&#xff0c;如果识别有问题&#xff0c;那么站点根本不会响应&#xff0c;所以为了完全模拟浏览器的工作&#xff0c;我们需要设置一些 Headers 的属性。 首先&#xff0c;打开我们的浏览器&#xff0c;调试浏览器…

进程上下文、中断上下文及原子上下文

谈论进程上下文 、中断上下文 、 原子上下文之前&#xff0c;有必要讨论下两个概念&#xff1a; a -- 上下文 上下文是从英文context翻译过来&#xff0c;指的是一种环境。相对于进程而言&#xff0c;就是进程执行时的环境&#xff1b; 具体来说就是各个变量和数据&#xff0c;…

Linux 文件系统与设备文件系统 (二)—— sysfs 文件系统与Linux设备模型

提到 sysfs 文件系统 &#xff0c;必须先需要了解的是Linux设备模型&#xff0c;什么是Linux设备模型呢&#xff1f; 一、Linux 设备模型 1、设备模型概述 从2.6版本开始&#xff0c;Linux开发团队便为内核建立起一个统一的设备模型。在以前的内核中没有独立的数据结构用来让内…

Python爬虫入门七正则表达式

已经搞定了怎样获取页面的内容&#xff0c;不过还差一步&#xff0c;这么多杂乱的代码夹杂文字我们怎样把它提取出来整理呢&#xff1f;下面就开始介绍一个十分强大的工具&#xff0c;正则表达式 1.了解正则表达式 正则表达式是用来匹配字符串非常强大的工具&#xff0c;在其…

Linux 文件系统与设备文件系统 (一)—— udev 设备文件系统

一、什么是Linux设备文件系统 首先我们不看定义&#xff0c;定义总是太抽象很难理解&#xff0c;我们先看现象。当我们往开发板上移植了一个新的文件系统之后&#xff08;假如各种设备驱动也移植好了&#xff09;&#xff0c;启动开发板&#xff0c;我们用串口工具进入开发板&a…

情人节,教大家使用css画出一朵玫瑰花。

情人节到了&#xff0c;给大家来一朵高端的玫瑰花。 在网上看到的一个canvas实现的玫瑰花&#xff0c;效果很好&#xff0c;但是代码被压缩过&#xff0c;也没有注释&#xff0c;看的云里雾里的。 今天我教大脚用CSS来实现一朵玫瑰花。 先看效果 首先我们画出一个花瓣 1、画出一…

Linux 字符设备驱动开发基础(六)—— VFS 虚拟文件系统解析

一、VFS 虚拟文件系统基础概念 Linux 允许众多不同的文件系统共存&#xff0c;并支持跨文件系统的文件操作&#xff0c;这是因为有虚拟文件系统的存在。虚拟文件系统&#xff0c;即VFS&#xff08;Virtual File System&#xff09;是 Linux 内核中的一个软件抽象层。它通过一些…

vim使用—实现程序的自动补齐(C语言)

使用过Source Insight的人一定对它的自动补全功能印象深刻&#xff0c;在很多的集成开发环境中&#xff0c;也都支持自动补全。vim做为一个出色的编辑器&#xff0c;这样的功能当然少不了。至于如何实现程序自动补全&#xff0c;网上教程很多。这里&#xff0c;我将自己配置过程…

[C#]Attribute特性(3)——AttributeUsage特性和特性标识符

相关文章 [C#]Attribute特性 [C#]Attribute特性(2)——方法的特性及特性参数 AttributeUsage特性 除了可以定制自己的特性来注释常用的C#类型外&#xff0c;您可以用AttributeUsage特性来定义您想怎样使用这些特性。AttributeUsage特性采用如下的调用惯例&#xff1a; 1 [Attri…

Linux 命令 ——less命令

less 工具也是对文件或其它输出进行分页显示的工具&#xff0c;应该说是linux正统查看文件内容的工具&#xff0c;功能极其强大。less 的用法比起 more 更加的有弹性。在 more 的时候&#xff0c;我们并没有办法向前面翻&#xff0c; 只能往后面看&#xff0c;但若使用了 less …

android闹钟实现原理

闹钟的原理可用下面我自己画的一幅图来概括&#xff1a;&#xff08;不对的地方&#xff0c;尽管吐槽&#xff09; 我们来看看新建闹钟到闹钟响铃的步骤&#xff1a; 1、新建一个闹钟&#xff1a; ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22…

将openstack的Token认证信息存储在memcache中

公司线上的openstack环境运行了5个多月的时间&#xff0c;keystone库的token表已经增长到了31GB&#xff0c;这个数据量还是很大的&#xff0c;对于以后的数据库备份很不方便。每次管理openstack的时候&#xff0c;都会产生一个新的token验证&#xff0c;而历史token信息其实都…

Linux 下shell编程

什么是shell?Shell是一个命令解析器&#xff0c;是介于Linux操作系统的内核(kernel)与用户之间的一个绝缘层。shell脚本就是讲各类命令预先放入其中&#xff0c;方便一次性执行的一个程序文件&#xff0c;主要用于方便管理员进行设置或者管理。 序员的角度来看&#xff0c; Sh…

linux 目录/sys 解析

今天学习Linux目录时&#xff0c;遇到/sys这个目录&#xff0c;老师怎么讲的&#xff0c;不太清楚&#xff0c;先对/sys目录知识进行一个整理 首先&#xff0c;对 /sys目录下的各个子目录进行具体说明&#xff1a; /sys下的子目录 内容 /sys/devices 该目录下…