嵌入式Linux之使用設(shè)備樹驅(qū)動GPIO的實現(xiàn)方式
在嵌入式 Linux 開發(fā)中,設(shè)備樹(Device Tree)和 GPIO 子系統(tǒng)是控制硬件設(shè)備的重要工具。
本文將詳細介紹如何使用設(shè)備樹和 GPIO 子系統(tǒng)驅(qū)動 LED 燈,包括在設(shè)備樹中添加 pinctrl 節(jié)點、設(shè)備節(jié)點,以及編寫驅(qū)動程序的全流程。
一、設(shè)備樹配置
1.1 添加 pinctrl 節(jié)點
pinctrl 節(jié)點用于配置 GPIO 引腳的功能和電氣屬性。
在實際開發(fā)中,通常在設(shè)備樹文件中,即dts文件中添加對應(yīng)的pinctrl節(jié)點來對對應(yīng)GPIO引腳的屬性進行定義。
如下即為對GPIO1_03引腳配置為GPIO模式等。

1.2 添加 LED 設(shè)備節(jié)點
在設(shè)備樹中添加 LED 設(shè)備節(jié)點,以描述 LED 的 GPIO 引腳和屬性,如下即為在設(shè)備樹根節(jié)點下添加一個gpioled設(shè)備節(jié)點,其中引用了定義的pinctrl節(jié)點中定義的屬性。

二、編寫驅(qū)動程序
2.1 驅(qū)動程序框架
在驅(qū)動文件中,主要完成對gpioled_dev結(jié)構(gòu)體的定義,四大文件操作函數(shù)的定義,初始化函數(shù)和退出函數(shù)的定義,起終點在于初始化函數(shù)的定義,如下幾位初始化函數(shù)的定義:
static int __init led_init(void)
{
int ret = 0;
/* 設(shè)置LED所使用的GPIO */
/* 1、獲取設(shè)備節(jié)點:gpioled */
gpioled.nd = of_find_node_by_path("/gpioled");//在設(shè)備樹中查找gpioled節(jié)點
if(gpioled.nd == NULL) {
printk("gpioled node not find!\r\n");
return -EINVAL;
} else {
printk("gpioled node find!\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");
}
/* 注冊字符設(shè)備驅(qū)動 */
/* 1、創(chuàng)建設(shè)備號 */
if (gpioled.major) { /* 定義了設(shè)備號 */
gpioled.devid = MKDEV(gpioled.major, 0);
register_chrdev_region(gpioled.devid, GPIOLED_CNT, GPIOLED_NAME);
} else { /* 沒有定義設(shè)備號 */
alloc_chrdev_region(&gpioled.devid, 0, GPIOLED_CNT, GPIOLED_NAME); /* 申請設(shè)備號 */
gpioled.major = MAJOR(gpioled.devid); /* 獲取分配號的主設(shè)備號 */
gpioled.minor = MINOR(gpioled.devid); /* 獲取分配號的次設(shè)備號 */
}
printk("gpioled major=%d,minor=%d\r\n",gpioled.major, gpioled.minor);
/* 2、初始化cdev */
gpioled.cdev.owner = THIS_MODULE;//與定義的gpioled_fops關(guān)聯(lián)在一起
cdev_init(&gpioled.cdev, &gpioled_fops);
/* 3、添加一個cdev */
cdev_add(&gpioled.cdev, gpioled.devid, GPIOLED_CNT);//添加到內(nèi)核
/* 4、創(chuàng)建類 */
gpioled.class = class_create(THIS_MODULE, GPIOLED_NAME);
if (IS_ERR(gpioled.class)) {
return PTR_ERR(gpioled.class);
}//創(chuàng)建類和設(shè)備節(jié)點,使用戶得以和內(nèi)核交互
/* 5、創(chuàng)建設(shè)備 */
gpioled.device = device_create(gpioled.class, NULL, gpioled.devid, NULL, GPIOLED_NAME);
if (IS_ERR(gpioled.device)) {
return PTR_ERR(gpioled.device);
}
return 0;
}即設(shè)置并注冊一個控制LED的字符設(shè)備驅(qū)動初始化函數(shù),首先查找設(shè)備樹節(jié)點,隨后在設(shè)備樹節(jié)點中獲取LED的GPIO編號,配置對應(yīng)的GPIO。以及注冊字符設(shè)備驅(qū)動,創(chuàng)建類和設(shè)備節(jié)點。
總體來說,即實現(xiàn)了從解析設(shè)備樹、配置GPIO、注冊字符設(shè)備驅(qū)動,到最后創(chuàng)建可被用戶空間程序訪問的設(shè)備節(jié)點。
這些步驟確保了驅(qū)動程序能夠正確地初始化硬件,并提供了一個與之交互的接口,讓用戶空間的應(yīng)用程序可以控制LED的狀態(tài)。
2.2 編譯驅(qū)動程序
使用make命令交叉編譯編寫的驅(qū)動文件,

隨后將編譯出的內(nèi)核模塊文件加載內(nèi)核中,即.ko文件:

三、測試
在嵌入式linux設(shè)備端,查看設(shè)備樹中添加的gpioled節(jié)點:

加載驅(qū)動,使用modprobe命令將加載gpioled.ko

最后,使用測試文件對驅(qū)動進行測試,觀察開發(fā)板狀態(tài),成功實現(xiàn)通過pinctrl和gpio子系統(tǒng)來完成對嵌入式linux系統(tǒng)中的GPIO進行控制。

總結(jié)
通過設(shè)備樹和 GPIO 子系統(tǒng)驅(qū)動 LED 燈的流程如下:
- 1. 在設(shè)備樹中添加 pinctrl 節(jié)點,配置 GPIO 引腳。
- 2. 在設(shè)備樹中添加 LED 設(shè)備節(jié)點,描述 LED 的 GPIO 引腳和屬性。
- 3. 編寫驅(qū)動程序,使用 GPIO 子系統(tǒng)控制 LED 燈。
- 4. 編譯并加載驅(qū)動,測試 LED 功能。
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Linux bash:./xxx:無法執(zhí)行二進制文件報錯
這篇文章主要介紹了Linux bash:./xxx:無法執(zhí)行二進制文件報錯,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-03-03
centos 5 手動分區(qū)來安裝系統(tǒng)的方法
近來我想體驗一下centos系統(tǒng),于是在虛擬機中安裝了centos 5。2010-03-03

