女人自慰AV免费观看内涵网,日韩国产剧情在线观看网址,神马电影网特片网,最新一级电影欧美,在线观看亚洲欧美日韩,黄色视频在线播放免费观看,ABO涨奶期羡澄,第一导航fulione,美女主播操b

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

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

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

pinctrl與gpio子系統(tǒng)下的字符設(shè)備驅(qū)動框架

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

pinctrl與gpio子系統(tǒng)下的字符設(shè)備驅(qū)動框架

點亮Linux驅(qū)動開發(fā)路上的第一個燈一文中將與外設(shè)有關(guān)的寄存器信息,定義到驅(qū)動代碼中,直接操作寄存器來控制外設(shè)。缺點是當芯片的寄存器發(fā)了變動,就要對底層的驅(qū)動進行重寫。

設(shè)備樹下的字符設(shè)備驅(qū)動框架一文中將與外設(shè)有關(guān)的寄存器信息,寫到了設(shè)備樹文件中,通過設(shè)備樹API函數(shù)獲取外設(shè)信息。當外設(shè)的信息有變化時,只需要修改設(shè)備樹文件即可,無需修改底層驅(qū)動,提高了驅(qū)動代碼的復(fù)用能力,但仍需要直接操作寄存器來控制外設(shè)。

本文介紹的pinctrl和gpio子系統(tǒng)實現(xiàn)了對寄存器的操作,我們只需要使用子系統(tǒng)提供的API函數(shù)即可,而無需再直接操作寄存器了。

1. pinctrl與gpio子系統(tǒng)介紹

1.1 pinctrl子系統(tǒng)

pinctrl子系統(tǒng)就是管理PIN引腳的一個系統(tǒng),在設(shè)備樹里設(shè)置PIN的配置信息,pinctrl會根據(jù)提供的信息來配置PIN功能。其主要工作內(nèi)容如下:

  • 獲取設(shè)備樹中的PIN信息
  • 設(shè)置獲取到的PIN的復(fù)用功能
  • 設(shè)置獲取到的PIN的電氣特性

打開設(shè)備樹文件imx6ull-andyxi-emmc.dts,其中iomuxc節(jié)點就是I.MX6ULL的外設(shè)節(jié)點。代碼中的pinctrl_hog_1子節(jié)點就是和熱插拔有關(guān)的PIN集合

&iomuxc {
    pinctrl-names = "default";
    pinctrl-0 = <&pinctrl_hog_1>;
    imx6ul-evk {
        pinctrl_hog_1: hoggrp-1 {
            fsl,pins = <
                MX6UL_PAD_UART1_RTS_B__GPIO1_IO19        0x17059 
                MX6UL_PAD_GPIO1_IO05__USDHC1_VSELECT     0x17059 
                MX6UL_PAD_GPIO1_IO09__GPIO1_IO09         0x17059 
                MX6UL_PAD_GPIO1_IO00__ANATOP_OTG1_ID     0x13058
           >;
        };
        ......
    };
};

下面以UART1_RTS_B這個PIN為例,介紹如何添加PIN的配置信息:

MX6UL_PAD_UART1_RTS_B__GPIO1_IO19     0x17059

前半部分:設(shè)置UART1_RTS_B引腳的復(fù)用功能

//MX6UL_PAD_UART1_RTS_B__GPIO1_IO19宏定義在文件arch/arm/boot/dts/imx6ul-pinfunc.h中
#define MX6UL_PAD_UART1_RTS_B__UART1_DCE_RTS          0x0090 0x031C 0x0620 0x0 0x3
#define MX6UL_PAD_UART1_RTS_B__UART1_DTE_CTS          0x0090 0x031C 0x0000 0x0 0x0
#define MX6UL_PAD_UART1_RTS_B__ENET1_TX_ER            0x0090 0x031C 0x0000 0x1 0x0
#define MX6UL_PAD_UART1_RTS_B__USDHC1_CD_B            0x0090 0x031C 0x0668 0x2 0x1
#define MX6UL_PAD_UART1_RTS_B__CSI_DATA05             0x0090 0x031C 0x04CC 0x3 0x1
#define MX6UL_PAD_UART1_RTS_B__ENET2_1588_EVENT1_OUT  0x0090 0x031C 0x0000 0x4 0x0
#define MX6UL_PAD_UART1_RTS_B__GPIO1_IO19             0x0090 0x031C 0x0000 0x5 0x0
#define MX6UL_PAD_UART1_RTS_B__USDHC2_CD_B            0x0090 0x031C 0x0674 0x8 0x2

上面代碼中共有8個UART1_RTS_B引腳的宏定義,分別對應(yīng)該引腳的8個復(fù)用IO。宏定義的值,被分為了5段,每段的值都有具體的含義:

圖片

  • mux_reg:mux_reg寄存器偏移地址 (見下圖1)
  • conf_reg:conf_reg寄存器偏移地址 (見下圖2)
  • input_reg:input_reg寄存器偏移地址 (此處無效)
  • mux_mode:mux_reg寄存器的值,用于設(shè)置復(fù)用功能 (見下圖3)
  • input_val:input_reg寄存器值 (此處無效)

圖片

圖片

圖片

后半部分:此值就是conf_reg寄存器的值,用于設(shè)置UART1_RTS_B引腳的電氣特性

1.2 gpio子系統(tǒng)

gpio子系統(tǒng)就是管理gpio功能的一個系統(tǒng),其作用是初始化gpio,并提供對外的API接口。使用gpio子系統(tǒng)后,就無需自己操作寄存器,通過調(diào)用相關(guān)API函數(shù)即可實現(xiàn)對gpio的控制。

仍以熱插拔節(jié)點為例,pinctrl子系統(tǒng)已經(jīng)將UART1_RTS_B復(fù)用為GPIO1_IO19,并設(shè)置好了電氣屬性。驅(qū)動程序通過讀取其高低電平來判斷SD卡有沒有插入。

那驅(qū)動程序怎么知道CD引腳連接的是GPIO1_IO19呢 ?這就需要設(shè)備樹來告訴驅(qū)動了,在設(shè)備樹的SD卡節(jié)點下添加一個描述CD引腳的屬性:

&usdhc1 {
    pinctrl-names = "default", "state_100mhz", "state_200mhz";
    pinctrl-0 = <&pinctrl_usdhc1>;
    pinctrl-1 = <&pinctrl_usdhc1_100mhz>;
    pinctrl-2 = <&pinctrl_usdhc1_200mhz>;
    pinctrl-3 = <&pinctrl_hog_1>;
    cd-gpios = <&gpio1 19 GPIO_ACTIVE_LOW>;
    keep-power-in-suspend;
    enable-sdio-wakeup;
    vmmc-supply = <®_sd1_vmmc>;
    status = "okay";
};

上面代碼中pinctrl-3指定了CD引腳的pinctrl信息,cd-gpios屬性描述了CD引腳使用哪個IO,屬性值共有三個,具體含義如下:

  • &gpio1 表示CD引腳所使用的IO屬于GPIO1組
  • 19 表示GPIO1組的第19號IO
  • GPIO_ACTIVE_LOW 表示低電平有效

設(shè)置好設(shè)備樹后就可使用gpio子系統(tǒng)提供的API函數(shù)來操作指定的GPIO,gpio子系統(tǒng)向開發(fā)人員屏蔽了具體的讀寫寄存器過程。下圖中有常用的API函數(shù)介,此外還有pinctrl和gpio子系統(tǒng)的使用模板:

圖片

**2. **pinctrl與gpio子系統(tǒng)字符設(shè)備驅(qū)動框架

下圖為pinctrl與gpio子系統(tǒng)下的字符設(shè)備驅(qū)動框架:

圖片

接下來根據(jù)上面的框架圖,以驅(qū)動LED (GPIO1_IO03) 為例,分步介紹具體的代碼編寫流程

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

添加pinctrl節(jié)點:在iomuxc節(jié)點的imx6ul-evk子節(jié)點下創(chuàng)建pinctrl_led節(jié)點,復(fù)用GPIO1_IO03

pinctrl_led: ledgrp {
    fsl,pins = <
        MX6UL_PAD_GPIO1_IO03__GPIO1_IO03    0x10B0
    >;
};
//MX6UL_PAD_GPIO1_IO03__GPIO1_IO03 復(fù)用GPIO1_IO03
//0x10B0 設(shè)置pin的電氣特性

添加LED設(shè)備節(jié)點:在根節(jié)點下創(chuàng)建LED設(shè)備節(jié)點,指定對應(yīng)的pinctrl節(jié)點,指定所使用的GPIO

gpioled {
    #address-cells = <1>;
    #size-cells = <1>;
    compatible = "alpha-gpioled";
    pinctrl-names = "default";
    pinctrl-0 = <&pinctrl_led>;
    led-gpio = <&gpio1 3 GPIO_ACTIVE_LOW>;
    status = "okay";
};

檢查PIN是否沖突:檢查pinctrl和設(shè)備節(jié)點中指定的引腳有沒有被別的外設(shè)占用

//檢查GPIO_IO03這個PIN有沒有被其他的pinctrl節(jié)點使用
pinctrl_tsc: tscgrp {
    fsl,pins = <
        MX6UL_PAD_GPIO1_IO01__GPIO1_IO01 0xb0
        MX6UL_PAD_GPIO1_IO02__GPIO1_IO02 0xb0
        //GPIO_IO03被pinctrl_tsc節(jié)點占用,因此需要屏蔽掉
        /* MX6UL_PAD_GPIO1_IO03__GPIO1_IO03 0xb0 */ 
        MX6UL_PAD_GPIO1_IO04__GPIO1_IO04 0xb0
    >;
};
//檢查"gpio1 3"有沒有被其他設(shè)備節(jié)點占用
&tsc {
    pinctrl-names = "default";
    pinctrl-0 = <&pinctrl_tsc>;
    //"gpio1 3"被tsc設(shè)備節(jié)點占用,因此需要屏蔽掉
    /* xnur-gpio = <&gpio1 3 GPIO_ACTIVE_LOW>; */
    measure-delay-time = <0xffff>;
    pre-charge-time = <0xfff>;
    status = "okay";
};

編譯設(shè)備樹:使用make dtbs命令編譯設(shè)備樹,并使用該設(shè)備樹啟動Linux系統(tǒng)

#在內(nèi)核根目錄下
make dtbs                #編譯設(shè)備樹
#啟動Linux系統(tǒng)后
cd /proc/device-tree     #查看"gpioled"節(jié)點是否存在

2.2 編寫驅(qū)動程序

創(chuàng)建驅(qū)動程序文件gpioled.c,添加如下代碼

宏定義及設(shè)備結(jié)構(gòu)體定義

struct gpioled_dev{
    dev_t devid;                //設(shè)備號
    struct cdev cdev;           //cdev字符設(shè)備
    struct class *class;        //類
    struct device *device;      //設(shè)備
    int major;                  //主設(shè)備號
    int minor;                  //次設(shè)備號
    struct device_node *nd;     //設(shè)備節(jié)點
    int led_gpio;               //所使用的gpio編號
};

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

編寫設(shè)備操作函數(shù):write函數(shù)中,直接使用gpio子系統(tǒng)提供的API函數(shù)操作GPIO

static ssize_t led_write(struct file *filp, char __user *buf, size_t cnt, loff_t *offt){
    int retvalue;
    unsigned char databuf[1];
    unsigned char ledstat;
    struct gpioled_dev *dev = filp->private_data;
    retvalue = copy_from_user(databuf, buf, cnt); 
    if(retvalue < 0){
        printk("kernel write failed!\\r\\n");
        return -EFAULT;
    }

    ledstat = databuf[0];
    if(ledstat == LEDON){
        gpio_set_value(dev->led_gpio, 0);
    }else if(ledstat == LEDOFF){
        gpio_set_value(dev->led_gpio, 1);
    }
    return 0;
}

驅(qū)動入口函數(shù)中:獲取GPIO編號后初始化

/* 設(shè)置 LED 所使用的 GPIO */
/* 1、獲取設(shè)備節(jié)點:gpioled */
gpioled.nd = of_find_node_by_path("/gpioled");
if(gpioled.nd == NULL) {
    printk("gpioled node cant not found!\\r\\n");
    return -EINVAL;
} else {
    printk("gpioled node has been found!\\r\\n");
}
/* 2、 獲取設(shè)備樹中的 gpio 屬性,得到 LED 所使用的 LED 編號 */
gpioled.led_gpio = of_get_named_gpio(gpioled.nd, "led-gpio", 0);
if(gpioled.led_gpio < 0) {
    printk("can't get led-gpio");
    return -EINVAL;
}
printk("led-gpio num = %d\\r\\n", gpioled.led_gpio);
/* 3、設(shè)置 GPIO1_IO03 為輸出,并且輸出高電平,默認關(guān)閉 LED 燈 */
ret = gpio_direction_output(gpioled.led_gpio, 1);
if(ret < 0) {
    printk("can't set gpio!\\r\\n");
}

驅(qū)動入口函數(shù)中:注冊字符設(shè)備驅(qū)動,代碼與Linux點燈中一樣

驅(qū)動出口函數(shù)中:注銷設(shè)備驅(qū)動,刪除類和設(shè)備,代碼可參考Linux點燈一文

2.3 編寫測試程序

實現(xiàn)操作驅(qū)動文件對外設(shè)進行控制的功能。創(chuàng)建測試程序文件gpioledApp.c,代碼內(nèi)容與Linux點燈一文中的測試程序代碼一致,此處不再贅述

2.4 編譯測試

編譯驅(qū)動程序:當前目錄下創(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 := gpioled.o

build: kernel_modules

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

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

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

運行測試:啟動開發(fā)板后,加載驅(qū)動模塊,之后使用應(yīng)用程序測試驅(qū)動是否正常工作

depmod                         #第一次加載驅(qū)動的時候需要運行此命令
modprobe gpioled.ko            #加載驅(qū)動
./gpioledApp /dev/gpioled 1    #打開LED燈
./gpioledApp /dev/gpioled 0    #關(guān)閉LED燈
rmmod gpioled.ko               #卸載驅(qū)動模塊
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 寄存器
    +關(guān)注

    關(guān)注

    31

    文章

    5416

    瀏覽量

    123224
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11450

    瀏覽量

    212705
  • 子系統(tǒng)
    +關(guān)注

    關(guān)注

    0

    文章

    114

    瀏覽量

    12662
  • 字符
    +關(guān)注

    關(guān)注

    0

    文章

    234

    瀏覽量

    25464
  • GPIO
    +關(guān)注

    關(guān)注

    16

    文章

    1267

    瀏覽量

    53511
收藏 人收藏

    評論

    相關(guān)推薦
    熱點推薦

    一文搞懂Linux pinctrl/gpio子系統(tǒng)

    GPIO的寄存器操作。分享給剛剛接觸外設(shè)bsp的小伙伴們。當然后面有時間還會分享GPIO子系統(tǒng)框架pinctrl
    發(fā)表于 06-09 09:52 ?3292次閱讀

    飛凌嵌入式ElfBoard ELF 1板卡-PinctrlGPIO子系統(tǒng)Pinctrl子系統(tǒng)

    。 pinctrl子系統(tǒng)的主要目的是將引腳的配置和控制與設(shè)備驅(qū)動程序的實現(xiàn)分離開來,從而提高驅(qū)動程序的可移植性和可重用性。通過使用
    發(fā)表于 03-22 09:23

    嵌入式學(xué)習(xí)-飛凌嵌入式ElfBoard ELF 1板卡-PinctrlGPIO子系統(tǒng)Pinctrl子系統(tǒng)

    pinctrl子系統(tǒng)的主要目的是將引腳的配置和控制與設(shè)備驅(qū)動程序的實現(xiàn)分離開來,從而提高驅(qū)動程序的可移植性和可重用性。通過使用
    發(fā)表于 03-24 15:42

    「正點原子Linux連載」第四十五章 pinctrlgpio子系統(tǒng)實驗(一)

    ,驅(qū)動分離與分層其實就是按照面向?qū)ο缶幊痰脑O(shè)計思想而設(shè)計的設(shè)備驅(qū)動框架,關(guān)于驅(qū)動的分離與分層我們后面會講。本來
    發(fā)表于 03-19 14:58

    基于GPIO子系統(tǒng)的LED驅(qū)動程序分享

    Pinctrl 子系統(tǒng)把引腳的復(fù)用、配置抽出來,做成 Pinctrl 子系統(tǒng),給 GPIO、I2C 等模塊使用。讓我們在使用某個引腳功能時不
    發(fā)表于 12-16 07:16

    怎樣去使用linux的pintcrl和gpio子系統(tǒng)

    pinctrlgpio內(nèi)部的原理是如何實現(xiàn)的?怎樣去使用linux的pintcrl和gpio子系統(tǒng)呢?
    發(fā)表于 03-07 13:38

    RK3399開發(fā)板的pinctrlgpio子系統(tǒng)相關(guān)資料介紹

    驅(qū)動工程師只做驅(qū)動,應(yīng)用工程師專注做應(yīng)用。  linux的pintcrl和gpio子系統(tǒng)就類似于ST的“BSP庫”,但是linux的
    發(fā)表于 09-16 17:27

    更新 | 持續(xù)開源 迅為RK3568驅(qū)動指南第十一篇-pinctrl子系統(tǒng)

    篇 熱插拔 第11篇 pinctrl子系統(tǒng) 未完待續(xù),持續(xù)更新中... 視頻教程更新至十二期 第一期_驅(qū)動基礎(chǔ) 第二期_字符設(shè)備基礎(chǔ) 第三期
    發(fā)表于 10-18 11:12

    gpiopinctrl子系統(tǒng)的關(guān)系與區(qū)別

    gpiopinctrl 子系統(tǒng)在內(nèi)核里的使用率非常高,和嵌入式產(chǎn)品的關(guān)聯(lián)非常大。從這兩個子系統(tǒng)開始學(xué)習(xí)驅(qū)動開發(fā)是個不錯的入門選擇。
    的頭像 發(fā)表于 03-15 11:40 ?5499次閱讀

    嵌入式驅(qū)動開發(fā)兩大子系統(tǒng)的使用

    本文的關(guān)注點是 gpio driver --> gpio subsystem core -> gpio consumer 這一路徑,讀者如果想更深入地了解 pinctrl
    的頭像 發(fā)表于 03-15 13:41 ?2004次閱讀

    使用pinctrlgpio子系統(tǒng)實現(xiàn)LED燈驅(qū)動

    前邊已經(jīng)學(xué)了兩種點燈,本質(zhì)依然還是通過配置寄存器;在學(xué)習(xí)STM32的時候除了學(xué)習(xí)配置一寄存器,基本都是使用庫來開發(fā),那么在i.MX6ULL還使用寄存器開發(fā)明顯是不太適合,那么i.MX6ULL有更方便的開發(fā)呢,這篇就來學(xué)習(xí)一使用 pi
    的頭像 發(fā)表于 04-03 10:17 ?1700次閱讀

    RK3568pinctrlgpio 子系統(tǒng)詳解

    如果 pinctrl 子系統(tǒng)將 PIN 復(fù)用為 GPIO,那么接下來就要配置 gpio 子系統(tǒng),且 gp
    的頭像 發(fā)表于 12-20 10:22 ?3107次閱讀
    RK3568<b class='flag-5'>pinctrl</b> 和 <b class='flag-5'>gpio</b> <b class='flag-5'>子系統(tǒng)</b>詳解

    Linux中pinctrl操作GPIO只需要幾步

    pinctrl 子系統(tǒng) API pinctrl 子系統(tǒng)的 API 有很多,對于驅(qū)動工程師來說,pinct
    的頭像 發(fā)表于 09-27 17:24 ?4593次閱讀

    瑞芯微RK3568-iomuxc和pinctrl子系統(tǒng)初窺

    pinctrl子系統(tǒng)作用:從設(shè)備樹中獲取PIN的描述信息來設(shè)置PIN的復(fù)用和電氣屬性,PIN可復(fù)用為I2C、SPI、GPIOgpio
    發(fā)表于 12-20 10:10 ?54次下載

    RK3568驅(qū)動指南|第十二篇 GPIO子系統(tǒng)-第135章 GPIO子系統(tǒng)pinctrl子系統(tǒng)相結(jié)合實驗

    RK3568驅(qū)動指南|第十二篇 GPIO子系統(tǒng)-第135章 GPIO子系統(tǒng)pinctrl
    的頭像 發(fā)表于 05-23 13:47 ?182次閱讀
    RK3568<b class='flag-5'>驅(qū)動</b>指南|第十二篇 <b class='flag-5'>GPIO</b><b class='flag-5'>子系統(tǒng)</b>-第135章 <b class='flag-5'>GPIO</b><b class='flag-5'>子系統(tǒng)</b>與<b class='flag-5'>pinctrl</b><b class='flag-5'>子系統(tǒng)</b>相結(jié)合實驗