等待队列中存放的是被阻塞的进程。而唤醒进程,就是从等待队列中把进程放到运行列表中去等待运行。
1.声明等待队列:
2.probe中初始化等待队列:
3.等待队列中去睡眠,其中又分为有条件睡眠和无条件睡眠,一般使用有条件睡眠:
有条件休眠函数:wait_event_interruptible(queue, condition)
当condition为true(BOOL值)时,立即返回;否则让进程进入TASK_INTERRUPTIBLE状态睡眠,并挂在queue所指定的阻塞队列上。
上面实例当!pn54x_dev->irq_enabled为true时,立即返回,继续后面代码的执行,为false,则阻塞函数pn54x_dev_read。而wake_up作用则是将阻塞函数放入运行队列中。
其中的enable_irq(pn54x_dev->client->irq) 表示在运行状态时,若产生中断,则会上报,在休眠状态不会上报。
还有一个enable_irq_wake(),它不仅仅在运行状态,而且在休眠状态时,产生的中断,也会上报到系统,这个多用在触摸屏唤醒系统。
4.wake_up唤醒等待队列中的进程,将其调度运行:
因此整个NFC代码流程就是:
当进程read时,发现!pn54x_dev->irq_enabled为false(pn54x_dev->irq_enabled = true导致),kernel将进程添加到阻塞队列(从运行队列中删除),一直循环等待条件为真(pn54x_dev->irq_enabled为false), 当检测到RF场范围内有TAG时,调用write函数将接收到的数据写入设备节点pn54x,此时有数据了,就会产生中断,产生高电平,也就是有数据可读时,触发中断进入中断处理函数pn54x_dev_irq_handler,而中断处理函数中将条件改变为 pn54x_dev->irq_enabled = false, 此时阻塞在wait_event_interruptible处的进程得以继续执行,再回到pn54x_dev_irq_handler中,此时wake_up就会将刚脱离阻塞的进程添加到运行队列。
天线调试好后才能读取到身份证信息,白卡,饭卡和交通卡比较容易读取到。