成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Linux驅動實踐:如何編寫【 GPIO 】設備的驅動程序?

系統 Linux
我們一塊討論了:在 Linux 系統中,編寫字符設備驅動程序的基本框架,主要是從代碼流程和 API 函數這兩方面觸發。這篇文章,我們就以此為基礎,寫一個有實際應用功能的驅動程序。

[[437191]]

別人的經驗,我們的階梯!

大家好,我是道哥。

在前幾篇文章中,我們一塊討論了:在 Linux 系統中,編寫字符設備驅動程序的基本框架,主要是從代碼流程和 API 函數這兩方面觸發。

這篇文章,我們就以此為基礎,寫一個有實際應用功能的驅動程序:

  1. 在驅動程序中,初始化 GPIO 設備,自動創建設備節點;
  2. 在應用程序中,打開 GPIO 設備,并發送控制指令設置 GPIO 口的狀態;

示例程序目標

編寫一個驅動程序模塊:mygpio.ko。

當這個驅動模塊被加載的時候,在系統中創建一個 mygpio 類設備,并且在 /dev 目錄下,創建 4 個設備節點:

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

因為我們現在是在 x86 平臺上來模擬 GPIO 的控制操作,并沒有實際的 GPIO 硬件設備。

因此,在驅動代碼中,與硬件相關部分的代碼,使用宏 MYGPIO_HW_ENABLE 控制起來,并且在其中使用printk輸出打印信息來體現硬件的操作。

在應用程序中,可以分別打開以上這 4 個 GPIO 設備,并且通過發送控制指令,來設置 GPIO 的狀態。

編寫驅動程序

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

創建驅動目錄和驅動程序

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

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

  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 硬件相關宏定義 
  8. #define MYGPIO_HW_ENABLE 
  9.  
  10. // 設備名稱 
  11. #define MYGPIO_NAME     "mygpio" 
  12.  
  13. // 一共有4個 GPIO 口 
  14. #define MYGPIO_NUMBER       4 
  15.  
  16. // 設備類 
  17. static struct class *gpio_class; 
  18.  
  19. // 用來保存設備 
  20. struct cdev gpio_cdev[MYGPIO_NUMBER]; 
  21.  
  22. // 用來保存設備號 
  23. int gpio_major = 0; 
  24. int gpio_minor = 0; 
  25.  
  26. #ifdef MYGPIO_HW_ENABLE 
  27. // 硬件初始化函數,在驅動程序被加載的時候(gpio_driver_init)被調用 
  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. // 設置硬件GPIO的狀態,在控制GPIO的時候(gpio_ioctl)被調研 
  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. // 當應用程序打開設備的時候被調用 
  41. static int gpio_open(struct inode *inode, struct file *file) 
  42.      
  43.     printk("gpio_open is called. \n"); 
  44.     return 0;    
  45.  
  46. // 當應用程序控制GPIO的時候被調用 
  47. static long gpio_ioctl(struct file* file, unsigned int val, unsigned long gpio_no) 
  48.     printk("gpio_ioctl is called. \n"); 
  49.      
  50.     // 檢查設置的狀態值是否合法 
  51.     if (0 != val && 1 != val) 
  52.     { 
  53.         printk("val is NOT valid! \n"); 
  54.         return 0; 
  55.     } 
  56.  
  57.     // 檢查設備范圍是否合法 
  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.     // 動態申請設備號(嚴謹點的話,應該檢查函數返回值) 
  86.     alloc_chrdev_region(&num_dev, gpio_minor, MYGPIO_NUMBER, MYGPIO_NAME); 
  87.  
  88.     // 獲取主設備號 
  89.     gpio_major = MAJOR(num_dev); 
  90.     printk("gpio_major = %d. \n", gpio_major); 
  91.  
  92.     // 創建設備類 
  93.     gpio_class = class_create(THIS_MODULE, MYGPIO_NAME); 
  94.  
  95.     // 創建設備節點 
  96.     for (i = 0; i < MYGPIO_NUMBER; ++i) 
  97.     { 
  98.         // 設備號 
  99.         devno = MKDEV(gpio_major, gpio_minor + i); 
  100.          
  101.         // 初始化 cdev 結構 
  102.         cdev_init(&gpio_cdev[i], &gpio_ops); 
  103.  
  104.         // 注冊字符設備 
  105.         cdev_add(&gpio_cdev[i], devno, 1); 
  106.  
  107.         // 創建設備節點 
  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.     // 刪除設備和設備節點 
  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.     // 釋放設備類 
  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.     // 注銷設備號 
  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); 

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

比如:在這個宏定義控制下的三個與硬件相關的函數:

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

就是與GPIO硬件的初始化、釋放、狀態設置相關的操作。

代碼中的注釋已經比較完善了,結合前幾篇文章中的函數說明,還是比較容易理解的。

從代碼中可以看出:驅動程序使用 alloc_chrdev_region 函數,來動態注冊設備號,并且利用了 Linux 應用層中的 udev 服務,自動在 /dev 目錄下創建了設備節點。

另外還有一點:在上面示例代碼中,對設備的操作函數只實現了 open 和 ioctl 這兩個函數,這是根據實際的使用場景來決定的。

這個示例中,只演示了如何控制 GPIO 的狀態。

你也可以稍微補充一下,增加一個read函數,來讀取某個GPIO口的狀態。

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

創建 Makefile 文件

  1. $ touch Makefile 

內容如下:

  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 

編譯驅動模塊

  1. $ make 

得到驅動程序: mygpio.ko 。

加載驅動模塊

在加載驅動模塊之前,先來檢查一下系統中,幾個與驅動設備相關的地方。

先看一下 /dev 目錄下,目前還沒有設備節點( /dev/mygpio[0-3] )。

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

再來查看一下 /proc/devices 目錄下,也沒有 mygpio 設備的設備號。

  1. $ cat /proc/devices 

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

  1. $ sudo dmesg -c 

現在來加載驅動模塊,執行如下指令:

  1. $ sudo insmod mygpio.ko 

當驅動程序被加載的時候,通過 module_init( ) 注冊的函數 gpio_driver_init() 將會被執行,那么其中的打印信息就會輸出。

還是通過 dmesg 指令來查看驅動模塊的打印信息:

  1. $ dmesg 

可以看到:操作系統為這個設備分配的主設備號是 244,并且也打印了GPIO硬件的初始化函數的調用信息。

此時,驅動模塊已經被加載了!

來查看一下 /proc/devices 目錄下顯示的設備號:

  1. $ cat /proc/devices 

設備已經注冊了,主設備號是: 244 。

設備節點

由于在驅動程序的初始化函數中,使用 cdev_add 和 device_create 這兩個函數,自動創建設備節點。

所以,此時我們在 /dev 目錄下,就可以看到下面這4個設備節點:

現在,設備的驅動程序已經加載了,設備節點也被創建好了,應用程序就可以來控制 GPIO 硬件設備了。

應用程序

應用程序仍然放在 ~/tmp/App/ 目錄下。

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

文件內容如下:

  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個設備節點 
  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.         // 參數個數檢查 
  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.         // 參數合法性檢查 
  33.     assert(gpio_no < MY_GPIO_NUMBER); 
  34.     assert(0 == val || 1 == val); 
  35.  
  36.     // 打開 GPIO 設備 
  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 設備狀態 
  45.     ioctl(fd, val, gpio_no); 
  46.      
  47.     // 關閉設備 
  48.     close(fd); 

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

接下來就是編譯和測試了:

  1. $ gcc app_mygpio.c -o app_mygpio 

執行應用程序的時候,需要攜帶2個參數:GPIO 設備編號(0 ~ 3),設置的狀態值(0 或者 1)。

這里設置一下/dev/mygpio0這個設備,狀態設置為1:

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

如何確認/dev/mygpio0這個GPIO的狀態確實被設置為1了呢?當然是看 dmesg 指令的打印信息:

  1. $ dmesg 

通過以上打印信息可以看到:確實執行了【設置 mygpio0 的狀態為 1】的動作。

再繼續測試一下:設置 mygpio0 的狀態為 0:

  1. $ sudo ./app_mygpio 0 0 

當然了,設置其他幾個GPIO口的狀態,都是可以正確執行的!

卸載驅動模塊

卸載指令:

  1. $ sudo rmmod mygpio 

此時,/proc/devices 下主設備號 244 的 mygpio 已經不存在了.

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

可以看到:驅動程序中的 gpio_driver_exit( ) 被調用執行了。

并且,/dev 目錄下的 4 個設備節點,也被函數 device_destroy() 自動刪除了!

本文轉載自微信公眾號「IOT物聯網小鎮」

 

責任編輯:姜華 來源: IOT物聯網小鎮
相關推薦

2011-01-10 18:21:38

linux編寫程序

2021-11-22 08:14:23

Linux Linux驅動Linux 系統

2021-12-06 07:47:36

Linux 驅動程序Linux 系統

2009-12-07 09:39:04

Linux設備驅動硬件通信

2021-11-16 06:55:36

Linux字符設備

2018-11-26 08:45:29

Linux驅動程序命令

2013-10-31 16:29:10

Linux內核

2021-11-12 11:28:01

Linux 內核驅動Linux 系統

2011-01-06 16:29:08

linuxtasklet機制

2010-01-07 13:27:22

Linux驅動程序

2010-05-10 15:53:24

Unix系統

2010-04-19 10:28:43

Unix操作系統

2009-07-06 18:17:46

JDBC驅動程序

2011-04-22 17:29:37

Linux網卡

2018-11-19 10:15:26

Windows 10WiFi驅動程序

2010-04-19 10:18:30

Unix操作系統

2019-10-22 15:40:34

Windows 10驅動程序Windows

2023-09-06 15:31:19

GPIO鴻蒙

2009-10-23 10:25:27

驅動程序技巧

2011-04-07 12:45:31

AMDAndroid平板電腦
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 日韩免费一区二区 | 日韩中文字幕一区 | 成年人在线观看视频 | 日韩成人免费视频 | 懂色中文一区二区三区在线视频 | 欧美一区在线视频 | 国产乱码精品一品二品 | 国产精品一区二 | 在线观看亚洲 | 特黄特色大片免费视频观看 | 伊人中文字幕 | 中文字幕日韩一区二区 | 亚洲精品在线免费播放 | 91九色婷婷 | 成年人的视频免费观看 | 久久精品99国产精品 | 亚洲国产欧美在线 | 国产精品日韩欧美一区二区三区 | 伊人中文字幕 | 欧美二区在线 | 成人一区二区三区在线观看 | 玖玖久久 | 亚洲国产欧美一区二区三区久久 | 欧美国产一区二区 | 日日骚av | 最新国产视频 | 精品精品| 波多野结衣中文字幕一区二区三区 | 四色永久 | 羞羞视频在线观看 | 欧美二区在线 | 久久精品国产99国产精品 | 久久免费大片 | 中文字幕国产精品 | 国产91网站在线观看 | 韩日一区二区三区 | 国产亚洲欧美另类一区二区三区 | 精品中文在线 | 国产视频中文字幕 | 精品视频一区二区三区在线观看 | 精品久久一区 |