1. 程式人生 > >Linux核心中的機制學習總結

Linux核心中的機制學習總結

一、驅動中的poll機制

1.簡介:
select()和poll()系統呼叫的本質一樣,前者在 BSD UNIX 中引入的,後者在 System V 中引入的。 應用程式使用 select() 或 poll() 呼叫裝置驅動程式的 file_operations 的 poll() 函式。

2.實現
(1).初始化一個等待佇列頭
init_waitqueue_head(&dev->rx_wait/&dev->tx_wait);

(2).實現驅動中的poll()方法

static unsigned int printer_poll(struct file *fd, poll_table *wait)
{
    
struct printer_dev *dev = fd->private_data; unsigned long flags; int status = 0; mutex_lock(&dev->lock_printer_io); spin_lock_irqsave(&dev->lock, flags); setup_rx_reqs(dev); spin_unlock_irqrestore(&dev->lock, flags); mutex_unlock(
&dev->lock_printer_io); poll_wait(fd, &dev->rx_wait, wait); poll_wait(fd, &dev->tx_wait, wait); spin_lock_irqsave(&dev->lock, flags); if (likely(!list_empty(&dev->tx_reqs))) status |= POLLOUT | POLLWRNORM; //可寫 if (likely(dev->current_rx_bytes) || likely(!list_empty(&dev->rx_buffers))) status
|= POLLIN | POLLRDNORM; //可讀 spin_unlock_irqrestore(&dev->lock, flags); return status; }

(3)喚醒等待佇列
wake_up_interruptible(&dev->rx_wait/&dev->tx_wait);
poll_wait函式並不會導致休眠。上面的驅動函式 printer_poll 返回位掩碼,如果掩碼為0,則表示裝置不可讀。若核心獲取到 printer_poll() 返回的掩碼為0,知道裝置不可讀,此時select函式就會被阻塞,程序休眠,等待有資料時被喚醒。所以,在寫入資料後,需要喚醒等待

注意:
1.wait_event_interruptible(dev->rx_wait, (likely(!list_empty(&dev->rx_buffers))));是用來實現阻塞型IO的,wake_up_interruptible(&dev->rx_wait);是用來喚醒.
2.printer_poll 並不會導致休眠,程序阻塞是系統呼叫select造成的。
3.系統呼叫select的阻塞會導致printer_poll 被呼叫多次!!