linux驱动分离分层的概念
這個分離分層的概念和輸入子系統有點像,但不是完全一樣的。為什么會再弄一個這個模型出來我也沒有搞懂,現在我的學習還停留在把知識學懂的層面上。至于為什么會產生這種知識,現在我還無從解釋,還需時日成長。
這次先上代碼在解釋整體構架。
devic.c
#include <linux/module.h> #include <linux/version.h>#include <linux/init.h>#include <linux/kernel.h> #include <linux/types.h> #include <linux/interrupt.h> #include <linux/list.h> #include <linux/timer.h> #include <linux/init.h> #include <linux/serial_core.h> #include <linux/platform_device.h>static void led_release(struct device * dev) { }static struct resource led_resource[] = {[0] = {.start = 0x56000010, /* TQ2440的LED是GPB5,6,7,8, GPBCON地址是0x56000010 */.end = 0x56000010 + 8 - 1,.flags = IORESOURCE_MEM,},[1] = {.start = 5, /* LED1 */.end = 5,.flags = IORESOURCE_IRQ,} };//首先要聲明這么一個結構體 并且填充里面的一些東西 name是很重要的//它是兩個文件相互匹配的依據 第二總要的就是resource 這個是聲明你的硬件資源的
static struct platform_device led_dev = {.name = "myled",.id = -1,.num_resources = ARRAY_SIZE(led_resource),.resource = led_resource,.dev = { .release = led_release, },};static int led_dev_init(void) {platform_device_register(&led_dev);return 0; }static void led_dev_exit(void) {platform_device_unregister(&led_dev);return ; }module_init(led_dev_init); module_exit(led_dev_exit); MODULE_LICENSE("GPL");
?
?
drive.c
#include <linux/module.h> #include <linux/version.h>#include <linux/init.h> #include <linux/fs.h> #include <linux/interrupt.h> #include <linux/irq.h> #include <linux/sched.h> #include <linux/pm.h> #include <linux/sysctl.h> #include <linux/proc_fs.h> #include <linux/delay.h> #include <linux/platform_device.h> #include <linux/input.h> #include <linux/irq.h> #include <asm/uaccess.h> #include <asm/io.h>static struct class *cls; static struct class_device *dev; static volatile unsigned long *gpio_con; static volatile unsigned long *gpio_dat; static int pin; int major;static int led_open(struct inode *inode, struct file *file) {*gpio_con &= ~(0x3<<(pin*2));*gpio_con |= (0x1<<(pin*2));return 0; }static ssize_t led_write(struct file *file, const char __user *buf, size_t count, loff_t * ppos) {int val;//printk("first_drv_write\n"); copy_from_user(&val, buf, count); // copy_to_user();if (val == 1){// 點燈*gpio_dat &= ~(1<<pin);}else{// 滅燈*gpio_dat |= (1<<pin);}return 0; }static struct file_operations led_fops = {.owner = THIS_MODULE, /* 這是一個宏,推向編譯模塊時自動創建的__this_module變量 */.open = led_open, .write = led_write, };static int led_probe(struct platform_device *pdev) {printk("led_probe!\n");/*從這里獲取在dev文件里面注冊的資源*/struct resource * res;res = platform_get_resource(pdev, IORESOURCE_MEM, 0);gpio_con = ioremap(res->start, res->end - res->start + 1);gpio_dat = gpio_con + 1;res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);pin = res->start;/*回到前面的驅動程序,注冊字符設備驅動程序*/major = register_chrdev(0, "myled", &led_fops);cls = class_create(THIS_MODULE, "myled");dev = class_device_create(cls, NULL, MKDEV(major, 0), NULL, "led");return 0; }static int led_remove(struct platform_device *pdev) {printk("led_remove!\n");class_device_unregister(dev);class_destroy(cls);unregister_chrdev(major, "myled");iounmap(gpio_con);return 0; }///填充 這個結構體struct platform_driver led_drv = {.probe = led_probe,.remove = led_remove,.driver = {.name = "myled",} };static int led_drv_init(void) {platform_driver_register(&led_drv);return 0; }static void led_drv_exit(void) {platform_driver_unregister(&led_drv);return ; }module_init(led_drv_init); module_exit(led_drv_exit);MODULE_LICENSE("GPL");
首先? 這個模型是依賴于內核里面的platform.c這個文件的。
和輸入子系統相比??? 這個模型的兩邊都要我們自己寫。
先看看device這邊,這邊全部都是與硬件相關的代碼。
我們可以看到? 這個文件主要就是聲明了一個結構體? 然后填充結構體里面的一些值
最重要的就是兩個東西?? 一個name 一個就是resource
填充完了之后告訴給這個模型的老大???? 也就是platform.c這個文件啦??
?
再看drive這邊。? drive這邊也填充了一個結構體? 并且向老大注冊了這個結構體(platform_driver)
我們知道?? 如果匹配兩邊的name匹配上了之后就會調用drive里面的led_probe 函數
在看看led_probe函數:
在里面取出了再device那邊注冊的硬件資源?? 然后再做各種處理?
余下的就和一般的驅動程序沒什么區別啦
?
這說的也是一點皮毛?? 要深入的了解還是要看內核的代碼???:
platform.c
?
轉載于:https://www.cnblogs.com/pslfree/p/3355531.html
總結
以上是生活随笔為你收集整理的linux驱动分离分层的概念的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: fastq-dump 报错 解决方案
- 下一篇: 数据库去重查询问题详解