自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Linux驅(qū)動(dòng)實(shí)踐:如何編寫【 GPIO 】設(shè)備的驅(qū)動(dòng)程序?

系統(tǒng) Linux
我們一塊討論了:在 Linux 系統(tǒng)中,編寫字符設(shè)備驅(qū)動(dòng)程序的基本框架,主要是從代碼流程和 API 函數(shù)這兩方面觸發(fā)。這篇文章,我們就以此為基礎(chǔ),寫一個(gè)有實(shí)際應(yīng)用功能的驅(qū)動(dòng)程序。

[[437191]]

別人的經(jīng)驗(yàn),我們的階梯!

大家好,我是道哥。

在前幾篇文章中,我們一塊討論了:在 Linux 系統(tǒng)中,編寫字符設(shè)備驅(qū)動(dòng)程序的基本框架,主要是從代碼流程和 API 函數(shù)這兩方面觸發(fā)。

這篇文章,我們就以此為基礎(chǔ),寫一個(gè)有實(shí)際應(yīng)用功能的驅(qū)動(dòng)程序:

  1. 在驅(qū)動(dòng)程序中,初始化 GPIO 設(shè)備,自動(dòng)創(chuàng)建設(shè)備節(jié)點(diǎn);
  2. 在應(yīng)用程序中,打開 GPIO 設(shè)備,并發(fā)送控制指令設(shè)置 GPIO 口的狀態(tài);

示例程序目標(biāo)

編寫一個(gè)驅(qū)動(dòng)程序模塊:mygpio.ko。

當(dāng)這個(gè)驅(qū)動(dòng)模塊被加載的時(shí)候,在系統(tǒng)中創(chuàng)建一個(gè) mygpio 類設(shè)備,并且在 /dev 目錄下,創(chuàng)建 4 個(gè)設(shè)備節(jié)點(diǎn):

  • /dev/mygpio0
  • /dev/mygpio1
  • /dev/mygpio2
  • /dev/mygpio3

因?yàn)槲覀儸F(xiàn)在是在 x86 平臺(tái)上來模擬 GPIO 的控制操作,并沒有實(shí)際的 GPIO 硬件設(shè)備。

因此,在驅(qū)動(dòng)代碼中,與硬件相關(guān)部分的代碼,使用宏 MYGPIO_HW_ENABLE 控制起來,并且在其中使用printk輸出打印信息來體現(xiàn)硬件的操作。

在應(yīng)用程序中,可以分別打開以上這 4 個(gè) GPIO 設(shè)備,并且通過發(fā)送控制指令,來設(shè)置 GPIO 的狀態(tài)。

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

以下所有操作的工作目錄,都是與上一篇文章相同的,即:~/tmp/linux-4.15/drivers/。

創(chuàng)建驅(qū)動(dòng)目錄和驅(qū)動(dòng)程序

  1. $ cd linux-4.15/drivers/ 
  2. $ mkdir mygpio_driver 
  3. $ cd mygpio_driver 
  4. $ touch mygpio.c 

mygpio.c 文件的內(nèi)容如下(不需要手敲,文末有代碼下載鏈接):

  1. #include <linux/module.h> 
  2. #include <linux/kernel.h> 
  3. #include <linux/ctype.h> 
  4. #include <linux/device.h> 
  5. #include <linux/cdev.h> 
  6.  
  7. // GPIO 硬件相關(guān)宏定義 
  8. #define MYGPIO_HW_ENABLE 
  9.  
  10. // 設(shè)備名稱 
  11. #define MYGPIO_NAME     "mygpio" 
  12.  
  13. // 一共有4個(gè) GPIO 口 
  14. #define MYGPIO_NUMBER       4 
  15.  
  16. // 設(shè)備類 
  17. static struct class *gpio_class; 
  18.  
  19. // 用來保存設(shè)備 
  20. struct cdev gpio_cdev[MYGPIO_NUMBER]; 
  21.  
  22. // 用來保存設(shè)備號(hào) 
  23. int gpio_major = 0; 
  24. int gpio_minor = 0; 
  25.  
  26. #ifdef MYGPIO_HW_ENABLE 
  27. // 硬件初始化函數(shù),在驅(qū)動(dòng)程序被加載的時(shí)候(gpio_driver_init)被調(diào)用 
  28. static void gpio_hw_init(int gpio) 
  29.     printk("gpio_hw_init is called: %d. \n", gpio); 
  30.  
  31. // 硬件釋放 
  32. static void gpio_hw_release(int gpio) 
  33.     printk("gpio_hw_release is called: %d. \n", gpio); 
  34.  
  35. // 設(shè)置硬件GPIO的狀態(tài),在控制GPIO的時(shí)候(gpio_ioctl)被調(diào)研 
  36. static void gpio_hw_set(unsigned long gpio_no, unsigned int val) 
  37.     printk("gpio_hw_set is called. gpio_no = %ld, val = %d. \n", gpio_no, val); 
  38. #endif 
  39.  
  40. // 當(dāng)應(yīng)用程序打開設(shè)備的時(shí)候被調(diào)用 
  41. static int gpio_open(struct inode *inode, struct file *file) 
  42.      
  43.     printk("gpio_open is called. \n"); 
  44.     return 0;    
  45.  
  46. // 當(dāng)應(yīng)用程序控制GPIO的時(shí)候被調(diào)用 
  47. static long gpio_ioctl(struct file* file, unsigned int val, unsigned long gpio_no) 
  48.     printk("gpio_ioctl is called. \n"); 
  49.      
  50.     // 檢查設(shè)置的狀態(tài)值是否合法 
  51.     if (0 != val && 1 != val) 
  52.     { 
  53.         printk("val is NOT valid! \n"); 
  54.         return 0; 
  55.     } 
  56.  
  57.     // 檢查設(shè)備范圍是否合法 
  58.     if (gpio_no >= MYGPIO_NUMBER) 
  59.     { 
  60.         printk("dev_no is invalid! \n"); 
  61.         return 0; 
  62.     } 
  63.  
  64.     printk("set GPIO: %ld to %d. \n", gpio_no, val); 
  65.  
  66. #ifdef MYGPIO_HW_ENABLE 
  67.     // 操作 GPIO 硬件 
  68.     gpio_hw_set(gpio_no, val); 
  69. #endif 
  70.  
  71.     return 0; 
  72.  
  73. static const struct file_operations gpio_ops={ 
  74.     .owner = THIS_MODULE, 
  75.     .open  = gpio_open, 
  76.     .unlocked_ioctl = gpio_ioctl 
  77. }; 
  78.  
  79. static int __init gpio_driver_init(void) 
  80.     int i, devno; 
  81.     dev_t num_dev; 
  82.  
  83.     printk("gpio_driver_init is called. \n"); 
  84.  
  85.     // 動(dòng)態(tài)申請(qǐng)?jiān)O(shè)備號(hào)(嚴(yán)謹(jǐn)點(diǎn)的話,應(yīng)該檢查函數(shù)返回值) 
  86.     alloc_chrdev_region(&num_dev, gpio_minor, MYGPIO_NUMBER, MYGPIO_NAME); 
  87.  
  88.     // 獲取主設(shè)備號(hào) 
  89.     gpio_major = MAJOR(num_dev); 
  90.     printk("gpio_major = %d. \n", gpio_major); 
  91.  
  92.     // 創(chuàng)建設(shè)備類 
  93.     gpio_class = class_create(THIS_MODULE, MYGPIO_NAME); 
  94.  
  95.     // 創(chuàng)建設(shè)備節(jié)點(diǎn) 
  96.     for (i = 0; i < MYGPIO_NUMBER; ++i) 
  97.     { 
  98.         // 設(shè)備號(hào) 
  99.         devno = MKDEV(gpio_major, gpio_minor + i); 
  100.          
  101.         // 初始化 cdev 結(jié)構(gòu) 
  102.         cdev_init(&gpio_cdev[i], &gpio_ops); 
  103.  
  104.         // 注冊(cè)字符設(shè)備 
  105.         cdev_add(&gpio_cdev[i], devno, 1); 
  106.  
  107.         // 創(chuàng)建設(shè)備節(jié)點(diǎn) 
  108.         device_create(gpio_class, NULL, devno, NULL, MYGPIO_NAME"%d", i); 
  109.     } 
  110.  
  111. #ifdef MYGPIO_HW_ENABLE 
  112.     // 初始化 GPIO 硬件 
  113.     for (i = 0; i < MYGPIO_NUMBER; ++i) 
  114.     { 
  115.         gpio_hw_init(i); 
  116.     } 
  117. #endif 
  118.  
  119.     return 0; 
  120.  
  121. static void __exit gpio_driver_exit(void) 
  122.     int i; 
  123.     printk("gpio_driver_exit is called. \n"); 
  124.  
  125.     // 刪除設(shè)備和設(shè)備節(jié)點(diǎn) 
  126.     for (i = 0; i < MYGPIO_NUMBER; ++i) 
  127.     { 
  128.         cdev_del(&gpio_cdev[i]); 
  129.         device_destroy(gpio_class, MKDEV(gpio_major, gpio_minor + i)); 
  130.     } 
  131.  
  132.     // 釋放設(shè)備類 
  133.     class_destroy(gpio_class); 
  134.  
  135. #ifdef MYGPIO_HW_ENABLE 
  136.     // 釋放 GPIO 硬件 
  137.     for (i = 0; i < MYGPIO_NUMBER; ++i) 
  138.     { 
  139.         gpio_hw_release(i); 
  140.     } 
  141. #endif 
  142.  
  143.     // 注銷設(shè)備號(hào) 
  144.     unregister_chrdev_region(MKDEV(gpio_major, gpio_minor), MYGPIO_NUMBER); 
  145.  
  146. MODULE_LICENSE("GPL"); 
  147. module_init(gpio_driver_init); 
  148. module_exit(gpio_driver_exit); 

相對(duì)于前幾篇文章來說,上面的代碼稍微有一點(diǎn)點(diǎn)復(fù)雜,主要是多了宏定義 MYGPIO_HW_ENABLE 控制部分的代碼。

比如:在這個(gè)宏定義控制下的三個(gè)與硬件相關(guān)的函數(shù):

  • gpio_hw_init()
  • gpio_hw_release()
  • gpio_hw_set()

就是與GPIO硬件的初始化、釋放、狀態(tài)設(shè)置相關(guān)的操作。

代碼中的注釋已經(jīng)比較完善了,結(jié)合前幾篇文章中的函數(shù)說明,還是比較容易理解的。

從代碼中可以看出:驅(qū)動(dòng)程序使用 alloc_chrdev_region 函數(shù),來動(dòng)態(tài)注冊(cè)設(shè)備號(hào),并且利用了 Linux 應(yīng)用層中的 udev 服務(wù),自動(dòng)在 /dev 目錄下創(chuàng)建了設(shè)備節(jié)點(diǎn)。

另外還有一點(diǎn):在上面示例代碼中,對(duì)設(shè)備的操作函數(shù)只實(shí)現(xiàn)了 open 和 ioctl 這兩個(gè)函數(shù),這是根據(jù)實(shí)際的使用場(chǎng)景來決定的。

這個(gè)示例中,只演示了如何控制 GPIO 的狀態(tài)。

你也可以稍微補(bǔ)充一下,增加一個(gè)read函數(shù),來讀取某個(gè)GPIO口的狀態(tài)。

控制 GPIO 設(shè)備,使用 write 或者 ioctl 函數(shù)都可以達(dá)到目的,只是 ioctl 更靈活一些。

創(chuàng)建 Makefile 文件

  1. $ touch Makefile 

內(nèi)容如下:

  1. ifneq ($(KERNELRELEASE),) 
  2.     obj-m := mygpio.o 
  3. else 
  4.     KERNELDIR ?= /lib/modules/$(shell uname -r)/build 
  5.     PWD := $(shell pwd) 
  6. default
  7.     $(MAKE) -C $(KERNELDIR) M=$(PWD) modules 
  8. clean: 
  9.     $(MAKE) -C $(KERNEL_PATH) M=$(PWD) clean 
  10. endif 

編譯驅(qū)動(dòng)模塊

  1. $ make 

得到驅(qū)動(dòng)程序: mygpio.ko 。

加載驅(qū)動(dòng)模塊

在加載驅(qū)動(dòng)模塊之前,先來檢查一下系統(tǒng)中,幾個(gè)與驅(qū)動(dòng)設(shè)備相關(guān)的地方。

先看一下 /dev 目錄下,目前還沒有設(shè)備節(jié)點(diǎn)( /dev/mygpio[0-3] )。

  1. $ ls -l /dev/mygpio* 
  2. ls: cannot access '/dev/mygpio*'No such file or directory 

再來查看一下 /proc/devices 目錄下,也沒有 mygpio 設(shè)備的設(shè)備號(hào)。

  1. $ cat /proc/devices 

為了方便查看打印信息,把dmesg輸出信息清理一下:

  1. $ sudo dmesg -c 

現(xiàn)在來加載驅(qū)動(dòng)模塊,執(zhí)行如下指令:

  1. $ sudo insmod mygpio.ko 

當(dāng)驅(qū)動(dòng)程序被加載的時(shí)候,通過 module_init( ) 注冊(cè)的函數(shù) gpio_driver_init() 將會(huì)被執(zhí)行,那么其中的打印信息就會(huì)輸出。

還是通過 dmesg 指令來查看驅(qū)動(dòng)模塊的打印信息:

  1. $ dmesg 

可以看到:操作系統(tǒng)為這個(gè)設(shè)備分配的主設(shè)備號(hào)是 244,并且也打印了GPIO硬件的初始化函數(shù)的調(diào)用信息。

此時(shí),驅(qū)動(dòng)模塊已經(jīng)被加載了!

來查看一下 /proc/devices 目錄下顯示的設(shè)備號(hào):

  1. $ cat /proc/devices 

設(shè)備已經(jīng)注冊(cè)了,主設(shè)備號(hào)是: 244 。

設(shè)備節(jié)點(diǎn)

由于在驅(qū)動(dòng)程序的初始化函數(shù)中,使用 cdev_add 和 device_create 這兩個(gè)函數(shù),自動(dòng)創(chuàng)建設(shè)備節(jié)點(diǎn)。

所以,此時(shí)我們?cè)?/dev 目錄下,就可以看到下面這4個(gè)設(shè)備節(jié)點(diǎn):

現(xiàn)在,設(shè)備的驅(qū)動(dòng)程序已經(jīng)加載了,設(shè)備節(jié)點(diǎn)也被創(chuàng)建好了,應(yīng)用程序就可以來控制 GPIO 硬件設(shè)備了。

應(yīng)用程序

應(yīng)用程序仍然放在 ~/tmp/App/ 目錄下。

  1. $ mkdir ~/tmp/App/app_mygpio 
  2. $ cd ~/tmp/App/app_mygpio 
  3. $ touch app_mygpio.c 

文件內(nèi)容如下:

  1. #include <stdio.h> 
  2. #include <stdlib.h> 
  3. #include <unistd.h> 
  4. #include <assert.h> 
  5. #include <fcntl.h> 
  6. #include <sys/ioctl.h> 
  7.  
  8. #define MY_GPIO_NUMBER      4 
  9.  
  10. // 4個(gè)設(shè)備節(jié)點(diǎn) 
  11. char gpio_name[MY_GPIO_NUMBER][16] = { 
  12.     "/dev/mygpio0"
  13.     "/dev/mygpio1"
  14.     "/dev/mygpio2"
  15.     "/dev/mygpio3" 
  16. }; 
  17.  
  18.  
  19. int main(int argc, char *argv[]) 
  20.     int fd, gpio_no, val; 
  21.  
  22.         // 參數(shù)個(gè)數(shù)檢查 
  23.     if (3 != argc) 
  24.     { 
  25.         printf("Usage: ./app_gpio gpio_no value \n"); 
  26.         return -1; 
  27.     } 
  28.  
  29.     gpio_no = atoi(argv[1]); 
  30.     val = atoi(argv[2]); 
  31.  
  32.         // 參數(shù)合法性檢查 
  33.     assert(gpio_no < MY_GPIO_NUMBER); 
  34.     assert(0 == val || 1 == val); 
  35.  
  36.     // 打開 GPIO 設(shè)備 
  37.     if((fd = open(gpio_name[gpio_no], O_RDWR | O_NDELAY)) < 0){ 
  38.         printf("%s: open failed! \n", gpio_name[gpio_no]); 
  39.         return -1; 
  40.     } 
  41.  
  42.     printf("%s: open success! \n", gpio_name[gpio_no]); 
  43.  
  44.     // 控制 GPIO 設(shè)備狀態(tài) 
  45.     ioctl(fd, val, gpio_no); 
  46.      
  47.     // 關(guān)閉設(shè)備 
  48.     close(fd); 

以上代碼也不需要過多解釋,只要注意參數(shù)的順序即可。

接下來就是編譯和測(cè)試了:

  1. $ gcc app_mygpio.c -o app_mygpio 

執(zhí)行應(yīng)用程序的時(shí)候,需要攜帶2個(gè)參數(shù):GPIO 設(shè)備編號(hào)(0 ~ 3),設(shè)置的狀態(tài)值(0 或者 1)。

這里設(shè)置一下/dev/mygpio0這個(gè)設(shè)備,狀態(tài)設(shè)置為1:

  1. $ sudo ./app_mygpio 0 1 
  2. [sudo] password for xxx: <輸入用戶密碼> 
  3. /dev/mygpio0: open success! 

如何確認(rèn)/dev/mygpio0這個(gè)GPIO的狀態(tài)確實(shí)被設(shè)置為1了呢?當(dāng)然是看 dmesg 指令的打印信息:

  1. $ dmesg 

通過以上打印信息可以看到:確實(shí)執(zhí)行了【設(shè)置 mygpio0 的狀態(tài)為 1】的動(dòng)作。

再繼續(xù)測(cè)試一下:設(shè)置 mygpio0 的狀態(tài)為 0:

  1. $ sudo ./app_mygpio 0 0 

當(dāng)然了,設(shè)置其他幾個(gè)GPIO口的狀態(tài),都是可以正確執(zhí)行的!

卸載驅(qū)動(dòng)模塊

卸載指令:

  1. $ sudo rmmod mygpio 

此時(shí),/proc/devices 下主設(shè)備號(hào) 244 的 mygpio 已經(jīng)不存在了.

再來看一下 dmesg的打印信息:

可以看到:驅(qū)動(dòng)程序中的 gpio_driver_exit( ) 被調(diào)用執(zhí)行了。

并且,/dev 目錄下的 4 個(gè)設(shè)備節(jié)點(diǎn),也被函數(shù) device_destroy() 自動(dòng)刪除了!

本文轉(zhuǎn)載自微信公眾號(hào)「IOT物聯(lián)網(wǎng)小鎮(zhèn)」

 

責(zé)任編輯:姜華 來源: IOT物聯(lián)網(wǎng)小鎮(zhèn)
相關(guān)推薦

2011-01-10 18:21:38

linux編寫程序

2021-11-22 08:14:23

Linux Linux驅(qū)動(dòng)Linux 系統(tǒng)

2021-12-06 07:47:36

Linux 驅(qū)動(dòng)程序Linux 系統(tǒng)

2009-12-07 09:39:04

Linux設(shè)備驅(qū)動(dòng)硬件通信

2021-11-16 06:55:36

Linux字符設(shè)備

2018-11-26 08:45:29

Linux驅(qū)動(dòng)程序命令

2013-10-31 16:29:10

Linux內(nèi)核

2021-11-12 11:28:01

Linux 內(nèi)核驅(qū)動(dòng)Linux 系統(tǒng)

2011-01-06 16:29:08

linuxtasklet機(jī)制

2010-01-07 13:27:22

Linux驅(qū)動(dòng)程序

2010-05-10 15:53:24

Unix系統(tǒng)

2010-04-19 10:28:43

Unix操作系統(tǒng)

2011-04-22 17:29:37

Linux網(wǎng)卡

2009-07-06 18:17:46

JDBC驅(qū)動(dòng)程序

2010-04-19 10:18:30

Unix操作系統(tǒng)

2018-11-19 10:15:26

Windows 10WiFi驅(qū)動(dòng)程序

2019-10-22 15:40:34

Windows 10驅(qū)動(dòng)程序Windows

2023-09-06 15:31:19

GPIO鴻蒙

2009-10-23 10:25:27

驅(qū)動(dòng)程序技巧

2011-04-07 12:45:31

AMDAndroid平板電腦
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)