linux設備驅動第五篇:驅動中的併發與竟態

綜述

在上一篇介紹了linux驅動的調試方法,這一篇介紹一下在驅動編程中會遇到的併發和竟態以及如何處理併發和競爭。linux

首先什麼是併發與竟態呢?併發(concurrency)指的是多個執行單元同時、並行被執行。而併發的執行單元對共享資源(硬件資源和軟件上的全局、靜態變量)的訪問則容易致使競態(race conditions)。可能致使併發和竟態的狀況有:android

  • SMP(Symmetric Multi-Processing),對稱多處理結構。SMP是一種緊耦合、共享存儲的系統模型,它的特色是多個CPU使用共同的系統總線,所以可訪問共同的外設和存儲器。 程序員

  • 中斷。中斷可 打斷正在執行的進程,若中斷處理程序訪問進程正在訪問的資源,則競態也會發生。中斷也可能被新的更高優先級的中斷打斷,所以,多箇中斷之間也可能引發併發而致使競態。算法

  • 內核進程的搶佔。linux是可搶佔的,因此一個內核進程可能被另外一個高優先級的內核進程搶佔。若是兩個進程共同訪問共享資源,就會出現竟態。chrome

以上三種狀況只有SMP是真正意義上的並行,而其餘都是宏觀上的並行,微觀上的串行。但其都會引起對臨界共享區的競爭問題。而解決競態問題的途徑是保證對共享資源的互斥訪問,即一個執行單元在訪問共享資源的時候,其餘的執行單元被禁止訪問。那麼linux內核中如何作到對對共享資源的互斥訪問呢?在linux驅動編程中,經常使用的解決併發與竟態的手段有信號量與互斥鎖,Completions 機制,自旋鎖(spin lock),以及一些其餘的不使用鎖的實現方式。下面一一介紹。編程

信號量與互斥鎖

信號量其實就是一個整型值,其核心是一個想進入臨界區的進程將在相關信號量上調用 P; 若是信號量的值大於零, 這個值遞減 1 而且進程繼續. 相反,,若是信號量的值是 0 ( 或更小 ), 進程必須等待直到別人釋放信號量. 解鎖一個信號量經過調用 V 完成; 這個函數遞增信號量的值,,而且, 若是須要, 喚醒等待的進程。而當信號量的初始值爲1的時候,就變成了互斥鎖。微信

信號量的典型使用形式:數據結構

//聲明信號量
struct semaphore sem;

//初始化信號量
void sema_init(struct semaphore *sem, int val)
    //經常使用下面兩種形式
#define init_MUTEX(sem) sema_init(sem, 1)
#define init_MUTEX_LOCKED(sem) sema_init(sem, 0)
    //如下是初始化信號量的快捷方式,最經常使用的
DECLARE_MUTEX(name)    //初始化name的信號量爲1
DECLARE_MUTEX_LOCKED(name) //初始化信號量爲0

//經常使用操做
DECLARE_MUTEX(mount_sem);
down(&mount_sem); //獲取信號量
...
critical section    //臨界區
...
up(&mount_sem);    //釋放信號量

經常使用的down操做還有併發

// 相似down(),由於down()而進入休眠的進程不能被信號打斷,而由於down_interruptible()而進入休眠的進程能被信號打斷, 
// 信號也會致使該函數返回,此時返回值非0
int down_interruptible(struct semaphore *sem);
// 嘗試得到信號量sem,若當即得到,它就得到該信號量並返回0,不然,返回非0.它不會致使調用者睡眠,可在中斷上下文使用
int down_trylock(struct semaphore *sem);

Completions 機制

完成量(completion)提供了一種比信號量更好的同步機制,它用於一個執行單元等待另外一個執行單元執行完某事。
微信公衆平臺

</pre></div><div><pre name="code" class="cpp">// 定義完成量
struct completion my_completion;
 
// 初始化completion
init_completion(&my_completion);
 
// 定義和初始化快捷方式:
DECLEAR_COMPLETION(my_completion);
 
// 等待一個completion被喚醒
void wait_for_completion(struct completion *c);
 
// 喚醒完成量
void cmplete(struct completion *c);
void cmplete_all(struct completion *c);

自旋鎖

若一個進程要訪問臨界資源,測試鎖空閒,則進程得到這個鎖並繼續執行;若測試結果代表鎖扔被佔用,進程將在一個小的循環內重複「測試並設置」操做,進行所謂的「自旋」,等待自旋鎖持有者釋放這個鎖。自旋鎖與互斥鎖相似,可是互斥鎖不能用在可能睡眠的代碼中,而自旋鎖能夠用在可睡眠的代碼中,典型的應用是能夠用在中斷處理函數中。自旋鎖的相關操做:

// 定義自旋鎖 
spinlock_t spin; 
 
// 初始化自旋鎖
spin_lock_init(lock);
 
// 得到自旋鎖:若能當即得到鎖,它得到鎖並返回,不然,自旋,直到該鎖持有者釋放
spin_lock(lock); 
 
// 嘗試得到自旋鎖:若能當即得到鎖,它得到並返回真,不然當即返回假,再也不自旋
spin_trylock(lock); 
 
// 釋放自旋鎖: 與spin_lock(lock)和spin_trylock(lock)配對使用
spin_unlock(lock); 
 
  自旋鎖的使用:
// 定義一個自旋鎖
spinlock_t lock;
spin_lock_init(&lock);
 
spin_lock(&lock);  // 獲取自旋鎖,保護臨界區
...  // 臨界區
spin_unlock();  // 解鎖


自旋鎖持有期間內核的搶佔將被禁止。自旋鎖能夠保證臨界區不受別的CPU和本CPU內的搶佔進程打擾,可是獲得鎖的代碼路徑在執行臨界區的時候還可能受到中斷和底半部(BH)的影響。爲防止這種影響,須要用到自旋鎖的衍生:

spin_lock_irq() = spin_lock() + local_irq_disable()
spin_unlock_irq() = spin_unlock() + local_irq_enable()
spin_lock_irqsave() = spin_lock() + local_irq_save()
spin_unlock_irqrestore() = spin_unlock() + local_irq_restore()
spin_lock_bh() = spin_lock() + local_bh_disable()
spin_unlock_bh() = spin_unlock() + local_bh_enable()

其餘的一些選擇

以上是linux驅動編程中常常用到的鎖機制,下面講一些內核中其餘的一些實現。

不加鎖算法

有時, 你能夠從新打造你的算法來徹底避免加鎖的須要.。許多讀者/寫者狀況 -- 若是隻有一個寫者 -- 經常可以在這個方式下工做.。若是寫者當心使數據結構,由讀者所見的,是一直一致的,,有可能建立一個不加鎖的數據結構。在linux內核中就有一個通用的無鎖的環形緩衝實現,具體內容參考<linux/kfifo.h>。

原子變量與位操做

原子操做指的是在執行過程當中不會被別的代碼路徑所中斷的操做。原子變量與位操做都是原子操做。如下是其相關操做介紹。

// 設置原子變量的值
void atomic_set(atomic_t *v, int i);  // 設置原子變量的值爲i
atomic_t v = ATOMIC_INIT(0);  // 定義原子變量v,並初始化爲0
 
// 獲取原子變量的值
atomic_read(atomic_t *v);  // 返回原子變量的值
 
// 原子變量加/減
void atomic_add(int i, atomic_t *v);  // 原子變量加i
void atomic_sub(int i, atomic_t *v);  // 原子變量減i
 
// 原子變量自增/自減
void atomic_inc(atomic_t *v);  // 原子變量增長1
void atomic_dec(atomic_t *v);  // 原子變量減小1
 
// 操做並測試:對原子變量進行自增、自減和減操做後(沒有加)測試其是否爲0,爲0則返回true,不然返回false
int atomic_inc_and_test(atomic_t *v);
int atomic_dec_and_test(atomic_t *v);
int atomic_sub_and_test(int i, atomic_t *v);
 
// 操做並返回: 對原子變量進行加/減和自增/自減操做,並返回新的值
int atomic_add_return(int i, atomic_t *v);
int atomic_sub_return(int i, atomic_t *v);
int atomic_inc_return(atomic_t *v);
int atomic_dec_return(atomic_t *v);
  位原子操做:
// 設置位
void set_bit(nr, void *addr);  // 設置addr地址的第nr位,即將位寫1
 
// 清除位
void clear_bit(nr, void *addr);  // 清除addr地址的第nr位,即將位寫0
 
// 改變位
void change_bit(nr, void *addr);  // 對addr地址的第nr位取反
 
// 測試位
test_bit(nr, void *addr); // 返回addr地址的第nr位
 
// 測試並操做:等同於執行test_bit(nr, void *addr)後再執行xxx_bit(nr, void *addr)
int test_and_set_bit(nr, void *addr);
int test_and_clear_bit(nr, void *addr);
int test_and_change_bit(nr, void *addr);

seqlock(順序鎖)

使用seqlock鎖,讀執行單元不會被寫執行單元阻塞,即讀執行單元能夠在寫執行單元對被seqlock鎖保護的共享資源進行寫操做時仍然能夠繼續讀,而沒必要等待寫執行單元完成寫操做,寫執行單元也不須要等待全部讀執行單元完成讀操做纔去進行寫操做。寫執行單元之間還是互斥的。若讀操做期間,發生了寫操做,必須從新讀取數據。seqlock鎖必需要求被保護的共享資源不含有指針。

// 得到順序鎖
void write_seqlock(seqlock_t *sl);
int write_tryseqlock(seqlock_t *sl);
write_seqlock_irqsave(lock, flags)
write_seqlock_irq(lock)
write_seqlock_bh()
 
// 釋放順序鎖
void write_sequnlock(seqlock_t *sl);
write_sequnlock_irqrestore(lock, flags)
write_sequnlock_irq(lock)
write_sequnlock_bh()
 
// 寫執行單元使用順序鎖的模式以下:
write_seqlock(&seqlock_a);
...  // 寫操做代碼塊
write_sequnlock(&seqlock_a);
  讀執行單元操做:
// 讀開始:返回順序鎖sl當前順序號
unsigned read_seqbegin(const seqlock_t *sl);
read_seqbegin_irqsave(lock, flags)
 
// 重讀:讀執行單元在訪問完被順序鎖sl保護的共享資源後須要調用該函數來檢查,在讀訪問期間是否有寫操做。如有寫操做,重讀
int read_seqretry(const seqlock_t *sl, unsigned iv);
read_seqretry_irqrestore(lock, iv, flags)
 
// 讀執行單元使用順序鎖的模式以下:
do{
    seqnum = read_seqbegin(&seqlock_a);
    // 讀操做代碼塊 
    ...
}while(read_seqretry(&seqlock_a, seqnum));

讀取-拷貝-更新(RCU)

讀取-拷貝-更新(RCU) 是一個高級的互斥方法,在合適的時候能夠取得很是高的效率。RCU能夠看做讀寫鎖的高性能版本,相比讀寫鎖,RCU的優勢在於既容許多個讀執行單元同時訪問被保護的數據,又容許多個讀執行單元和多個寫執行單元同時訪問被保護的數據。可是RCU不能替代讀寫鎖,由於若是寫比較多時,對讀執行單元的性能提升不能彌補寫執行單元致使的損失。因爲平時應用較少,因此不作多說。

小結

以上就是linux驅動編程中涉及的併發與競態的內容,下面作一個簡單的小結。

如今的處理器基本上都是SMP類型的,並且在新的內核版本中,基本上都支持搶佔式的操做,在linux中不少程序都是可重入的,要保護這些數據,就得使用不一樣的鎖機制。而鎖機制的基本操做過程其實大同小異的,聲明變量,上鎖,執行臨界區代碼,而後再解鎖。不一樣點在於,能夠重入的限制不一樣,有的能夠無限制重入,有的只容許異種操做重入,而有的是不容許重入操做的,有的能夠在可睡眠代碼中使用,有的不能夠在可睡眠代碼中使用。而在考慮不一樣的鎖機制的使用時,也要考慮CPU處理的效率問題,對於不一樣的代碼長度,不一樣的代碼執行時間,選擇一個好的鎖對CPU的良好使用有很大的影響,不然將形成浪費。 

以前在linux設備驅動第三篇:寫一個簡單的字符設備驅動中介紹了簡單的字符設備驅動,下一篇將介紹一些字符設備驅動中得高級操做。

第一時間得到博客更新提醒,以及更多技術信息分享,歡迎關注我的微信公衆平臺:程序員互動聯盟(coder_online),掃一掃下方二維碼或搜索微信號coder_online便可關注,閱讀android,chrome等多種熱門技術文章。

相關文章
相關標籤/搜索