linux字元裝置驅動程式的設計之休眠
休眠對進程來說,意味著等待將來的某個時間發生。如何以安全的方式進入休眠,需要注意以下兩點:
永遠不要在原子上下文中進入休眠。
當進程被喚醒時,我們永遠無法知道休眠了多長時間,或者休眠期間發生了什麼事情。
等待隊列就是一個進程鏈表,其中包含了等待某個特定事件的所有進程。在linux中,一個等待隊列通過一個“等待隊列頭(wait quene head)”來管理,等待隊列頭是一個類型為wait_quene_head_t的結構體,定義在<linux/wait,h>中,可通過靜態定義並初始化一個等待隊列頭:
DECLARE_WAIT_QUENE_HEAD(name);
define DECLARE_WAIT_QUEUE_HEAD(name) \wait_queue_head_t name = __WAIT_QUEUE_HEAD_INITIALIZER(name)
或者使用動態方法:
wait_quene_head_t my_quene;
init_waitquene_head (&my_quene);
void __init_waitqueue_head(wait_queue_head_t *q, struct lock_class_key *key){spin_lock_init(&q->lock);lockdep_set_class(&q->lock, key);INIT_LIST_HEAD(&q->task_list);}
typedef struct __wait_queue_head wait_queue_head_t;
struct __wait_queue_head {spinlock_t lock;struct list_head task_list;};
<1>、簡單休眠
#define wait_event(wq, condition) \do { \ if (condition) \ break; \ __wait_event(wq, condition); \} while (0)
#define wait_event_timeout(wq, condition, timeout)\({\long __ret = timeout;\if (!(condition)) \__wait_event_timeout(wq, condition, __ret);\__ret;\})
#define wait_event_interruptible(wq, condition)\({\int __ret = 0;\if (!(condition))\__wait_event_interruptible(wq, condition, __ret);\__ret;\})
#define wait_event_interruptible_timeout(wq, condition, timeout)\({\long __ret = timeout;\if (!(condition))\__wait_event_interruptible_timeout(wq, condition, __ret); \__ret;\})
<2>喚醒休眠wake_up()
#define wake_up(x)__wake_up(x, TASK_NORMAL, 1, NULL)
#define wake_up_interruptible(x) __wake_up(x, TASK_INTERRUPTIBLE, 1, NULL)