0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

使用Linux互斥體實(shí)現(xiàn)互斥點(diǎn)燈

CHANBAEK ? 來源:嵌入式攻城獅 ? 作者:安迪西 ? 2023-04-13 15:13 ? 次閱讀

1. 互斥體介紹

互斥訪問是指一次只有一個(gè)線程可以訪問共享資源,不能遞歸申請(qǐng)互斥體。使用互斥體時(shí)要注意如下幾點(diǎn):

  • mutex可導(dǎo)致休眠,因此不能在中斷中使用
  • mutex保護(hù)的臨界區(qū)可調(diào)用引起阻塞的API函數(shù)
  • 必須由mutex的持有者釋放mutex,且不能遞歸上鎖和解鎖

Linux 內(nèi)核使用mutex結(jié)構(gòu)體表示互斥體,結(jié)構(gòu)體定義如下所示:

struct mutex {
    /* 1: unlocked, 0: locked, negative: locked, possible waiters */
    atomic_t count;
    spinlock_t wait_lock;
};

互斥體操作相關(guān)API函數(shù)如下圖示:

圖片

2. 互斥體實(shí)例

本實(shí)例中使用互斥體來實(shí)現(xiàn)對(duì)LED設(shè)備的互斥訪問,即一次只允許一個(gè)應(yīng)用程序使用LED燈,代碼是在pinctrl與gpio子系統(tǒng)下的字符設(shè)備驅(qū)動(dòng)框架一文基礎(chǔ)上完成的

2.1 修改設(shè)備樹文件

設(shè)備樹文件修改與pinctrl與gpio子系統(tǒng)下的字符設(shè)備驅(qū)動(dòng)框架文中的修改方法一樣,不需要做任何修改

2.2 編寫驅(qū)動(dòng)程序

拷貝pinctrl與gpio子系統(tǒng)下的字符設(shè)備驅(qū)動(dòng)框架文中的gpioled.c驅(qū)動(dòng)文件,并重命名為mutex.c,對(duì)部分代碼進(jìn)行修改,其余保持不變

在設(shè)備結(jié)構(gòu)體中,添加互斥體

struct gpioled_dev{
    dev_t devid;              //設(shè)備號(hào)
    struct cdev cdev;         //cdev字符設(shè)備
    struct class *class;      //類
    struct device *device;    //設(shè)備
    int major;                //主設(shè)備號(hào)
    int minor;                //次設(shè)備號(hào)
    struct device_node *nd;   //設(shè)備節(jié)點(diǎn)
    int led_gpio;             //所使用的gpio編號(hào)
    struct mutex lock;        //互斥體
};

struct gpioled_dev gpioled;   //定義led設(shè)備

打開設(shè)備時(shí),獲取互斥體

static int led_open(struct inode *inode, struct file *filp){
    filp->private_data = &gpioled; //設(shè)置私有數(shù)據(jù)

    if(mutex_lock_interruptible(&gpioled.lock)){
        return -ERESTARTSYS;       //進(jìn)入休眠后可被信號(hào)打斷
    } 
#if 0
    mutex_lock(&gpioled.lock);     //休眠后不能被信號(hào)打斷
#endif
    return 0;
}

關(guān)閉設(shè)備時(shí),釋放互斥鎖

static int led_release(struct inode *inode, struct file *filp){
    struct gpioled_dev *dev = filp->private_data;
    mutex_unlock(&dev->lock); //釋放互斥鎖
    return 0;
}

驅(qū)動(dòng)入口函數(shù)中,對(duì)互斥體進(jìn)行初始化

static int __init led_init(void){
    int ret = 0;
    /* 初始化互斥體 */
    mutex_init(&gpioled.lock);  
    /* 設(shè)置 LED 所使用的 GPIO */
    /* 1、獲取設(shè)備節(jié)點(diǎn):gpioled */
    gpioled.nd = of_find_node_by_path("/gpioled");
    ......
    ......
}

2.3 編寫測(cè)試程序

拷貝pinctrl與gpio子系統(tǒng)下的字符設(shè)備驅(qū)動(dòng)框架文中的gpioledApp.c測(cè)試程序,并重命名為mutexApp.c,添加模擬占用LED的代碼,使測(cè)試程序在獲取LED驅(qū)動(dòng)使用權(quán)后會(huì)持續(xù)一段時(shí)間,添加如下代碼

while(1){
    sleep(5);
    cnt++;
    printf("App running times: %d\\r\\n",cnt);
    if(cnt >= 5)
        break;
}

2.4 編譯測(cè)試

編譯驅(qū)動(dòng)程序:當(dāng)前目錄下創(chuàng)建Makefile文件,并使用make命令編譯

KERNELDIR := /home/andyxi/linux/kernel/linux-imx-rel_imx_4.1.15_2.1.0_ga_andyxi
CURRENT_PATH := $(shell pwd)
obj-m := mutex.o

build: kernel_modules

kernel_modules:
$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) modules
clean:
$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) clean

編譯測(cè)試程序:無需內(nèi)核參與,直接編譯即可

arm-linux-gnueabihf-gcc mutexApp.c -o mutexApp

運(yùn)行測(cè)試:?jiǎn)?dòng)開發(fā)板后,加載驅(qū)動(dòng)模塊,操作LED燈后,相應(yīng)時(shí)間內(nèi)再次操作LED時(shí),第二個(gè)應(yīng)用程序會(huì)進(jìn)入休眠,等前面的應(yīng)用程序運(yùn)行完后,第二個(gè)應(yīng)用程序會(huì)接著運(yùn)行

depmod                       #第一次加載驅(qū)動(dòng)的時(shí)候需要運(yùn)行此命令
modprobe mutex.ko            #加載驅(qū)動(dòng)
# 打開LED后,每隔5秒會(huì)輸出一行App running times
./mutexApp /dev/gpioled 1&   # &表示在后臺(tái)運(yùn)行APP

圖片

#在LED被占用期間,再次操作LED,會(huì)因?yàn)楂@取互斥體失敗而進(jìn)入休眠狀態(tài),
#等前一條命令運(yùn)行完并釋放互斥鎖后才能獲取LED使用權(quán)
./mutexApp /dev/gpioled 0

圖片

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • led
    led
    +關(guān)注

    關(guān)注

    242

    文章

    23286

    瀏覽量

    661112
  • 內(nèi)核
    +關(guān)注

    關(guān)注

    3

    文章

    1372

    瀏覽量

    40303
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11310

    瀏覽量

    209621
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4332

    瀏覽量

    62666
  • 線程
    +關(guān)注

    關(guān)注

    0

    文章

    505

    瀏覽量

    19695
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    詳細(xì)談?wù)?b class='flag-5'>Linux中的多線程同步和互斥機(jī)制

    互斥:多線程中互斥是指多個(gè)線程訪問同一資源時(shí)同時(shí)只允許一個(gè)線程對(duì)其進(jìn)行訪問,具有唯一性和排它性。但互斥無法限制訪問者對(duì)資源的訪問順序,即訪問是無序的;
    的頭像 發(fā)表于 03-20 09:09 ?1872次閱讀

    Linux高級(jí)編程---互斥

    Linux系統(tǒng)里,有很多鎖的應(yīng)用,包括互斥鎖,文件鎖,讀寫鎖等等,信號(hào)量其實(shí)也應(yīng)該是鎖的一種。使用鎖的目的是為了達(dá)到進(jìn)程、線程之間的同步作用,使共享資源在同一時(shí)間內(nèi),只有能有一個(gè)進(jìn)程或者線程對(duì)它
    發(fā)表于 01-13 10:07

    互斥量源碼分析測(cè)試

    文章目錄互斥量源碼分析測(cè)試參考資料:RTT官網(wǎng)文檔關(guān)鍵字:分析RT-Thread源碼、stm32、RTOS、互斥量。互斥量在其他書籍中的名稱:mutex :互斥鎖,
    發(fā)表于 08-24 06:01

    信號(hào)量和互斥鎖的區(qū)別

    互斥量用于線程的互斥,信號(hào)線用于線程的同步。這是互斥量和信號(hào)量的根本區(qū)別,也就是互斥和同步之間的區(qū)別。互斥:是指某一資源同時(shí)只允許一個(gè)訪問者
    發(fā)表于 11-13 17:43 ?1.3w次閱讀
    信號(hào)量和<b class='flag-5'>互斥</b>鎖的區(qū)別

    Linux 多線程互斥互斥

    的。如果操作是原子操作,那么天然的具有互斥同步:是指在互斥的基礎(chǔ)上(大多數(shù)情況),通過其它機(jī)制實(shí)現(xiàn)訪問者對(duì)資源的有序訪問。在大多數(shù)情況下,同步已經(jīng)實(shí)現(xiàn)
    發(fā)表于 04-02 14:47 ?272次閱讀

    詳談Linux操作系統(tǒng)編程的互斥量mutex

    前文提到,系統(tǒng)中如果存在資源共享,線程間存在競(jìng)爭(zhēng),并且沒有合理的同步機(jī)制的話,會(huì)出現(xiàn)數(shù)據(jù)混亂的現(xiàn)象。為了實(shí)現(xiàn)同步機(jī)制,Linux中提供了多種方式,其中一種方式為互斥鎖mutex(也稱之為互斥
    的頭像 發(fā)表于 09-28 15:09 ?2512次閱讀
    詳談<b class='flag-5'>Linux</b>操作系統(tǒng)編程的<b class='flag-5'>互斥</b>量mutex

    詳解互斥信號(hào)量的概念和運(yùn)行

    1 、互 斥 信 號(hào) 量 1.1 互斥信號(hào)量的概念及其作用 互斥信號(hào)量的主要作用是對(duì)資源實(shí)現(xiàn)互斥訪問,使用二值信號(hào)量也可以實(shí)現(xiàn)
    的頭像 發(fā)表于 10-22 11:57 ?1.2w次閱讀
    詳解<b class='flag-5'>互斥</b>信號(hào)量的概念和運(yùn)行

    Linux中的傷害/等待互斥鎖介紹

    序言:近期讀Linux 5.15的發(fā)布說明,該版本合并了實(shí)時(shí)鎖機(jī)制,當(dāng)開啟配置宏CONFIG_PREEMPT_RT的時(shí)候,這些鎖被基于實(shí)時(shí)互斥鎖的變體替代:mutex、ww_mutex
    的頭像 發(fā)表于 11-06 17:27 ?2674次閱讀

    Linux下線程間通訊--互斥

    互斥鎖是一種簡(jiǎn)單的加鎖的方法來控制對(duì)共享資源的存取,當(dāng)多個(gè)線程訪問公共資源時(shí),為了保證同一時(shí)刻只有一個(gè)線程獨(dú)占資源,就可以通過互斥鎖加以限制,在一個(gè)時(shí)刻只能有一個(gè)線程掌握某個(gè)互斥鎖,擁有上鎖狀態(tài)
    的頭像 發(fā)表于 08-24 15:53 ?1970次閱讀
    <b class='flag-5'>Linux</b>下線程間通訊--<b class='flag-5'>互斥</b>鎖

    Free RTOS的互斥信號(hào)量

    二進(jìn)制信號(hào)量和互斥量非常相似,但確實(shí)有一些細(xì)微的區(qū)別。互斥包含優(yōu)先級(jí)繼承機(jī)制,而二進(jìn)制信號(hào)量沒有。這使得二進(jìn)制信號(hào)量成為實(shí)現(xiàn)同步(任務(wù)之間或任務(wù)與中斷之間)的更好選擇,
    的頭像 發(fā)表于 02-10 15:36 ?1191次閱讀
    Free RTOS的<b class='flag-5'>互斥</b>信號(hào)量

    使用Linux原子操作實(shí)現(xiàn)互斥點(diǎn)燈

    Linux原子操作是指不能再進(jìn)一步分割的操作,一般原子操作用于變量或者位操作。Linux內(nèi)核定義了atomic_t結(jié)構(gòu)來完成整型數(shù)據(jù)的原子操作,此結(jié)構(gòu)定義在include/
    的頭像 發(fā)表于 04-13 15:07 ?892次閱讀
    使用<b class='flag-5'>Linux</b>原子操作<b class='flag-5'>實(shí)現(xiàn)</b><b class='flag-5'>互斥</b><b class='flag-5'>點(diǎn)燈</b>

    使用Linux信號(hào)量實(shí)現(xiàn)互斥點(diǎn)燈

    信號(hào)量常用于控制對(duì)共享資源的訪問,有計(jì)數(shù)型信號(hào)量和二值信號(hào)量之分。初始化時(shí)信號(hào)量值大于1的,就是計(jì)數(shù)型信號(hào)量,計(jì)數(shù)型信號(hào)量不能用于互斥訪問,它允許多個(gè)線程同時(shí)訪問共享資源。若要互斥訪問共享資源,信號(hào)量的值就不能大于1,此時(shí)就是二值信號(hào)量。
    的頭像 發(fā)表于 04-13 15:12 ?813次閱讀
    使用<b class='flag-5'>Linux</b>信號(hào)量<b class='flag-5'>實(shí)現(xiàn)</b><b class='flag-5'>互斥</b><b class='flag-5'>點(diǎn)燈</b>

    Linux互斥鎖的作用 互斥鎖是什么

    1、互斥互斥鎖(mutex),在訪問共享資源之前對(duì)互斥鎖進(jìn)行上鎖,在訪問完成后釋放互斥鎖(解鎖);對(duì)互斥鎖進(jìn)行上鎖之后,任何其它試圖再次
    的頭像 發(fā)表于 07-21 11:13 ?951次閱讀

    自旋鎖和互斥鎖的區(qū)別有哪些

    自旋鎖 自旋鎖與互斥鎖很相似,在訪問共享資源之前對(duì)自旋鎖進(jìn)行上鎖,在訪問完成后釋放自旋鎖(解鎖);事實(shí)上,從實(shí)現(xiàn)方式上來說,互斥鎖是基于自旋鎖來實(shí)現(xiàn)的,所以自旋鎖相較于
    的頭像 發(fā)表于 07-21 11:19 ?9506次閱讀

    互斥鎖和自旋鎖的實(shí)現(xiàn)原理

    保護(hù)共享資源不被多個(gè)線程同時(shí)訪問。它的實(shí)現(xiàn)原理主要包括以下幾個(gè)方面: 1. 鎖的初始化 互斥鎖在創(chuàng)建時(shí)需要進(jìn)行初始化,通常包括設(shè)置鎖的狀態(tài)為“未鎖定”。在某些實(shí)現(xiàn)中,還需要初始化鎖的等待隊(duì)列,用于存儲(chǔ)等待鎖的線程。 2. 鎖的
    的頭像 發(fā)表于 07-10 10:07 ?501次閱讀