嵌入式内核及驱动开发之学习笔记(十二) 设备驱动模型
按照之前的实现驱动的方法。对于同一类设备,驱动层实现的操作方法是相似的,而设备信息总是不同的,一个产品上有很多这样的同类设备,我们为每一个设备都定制一套驱动程序,代价似乎太“高昂点”;对于一个设别,如果硬件有所改动,也必然要修改取代代码。
为了提高代码的通用性,我们将驱动和设备进行分离,设备对象专门用来描述设备(硬件)的信息。而驱动负责重设备中获取这些硬件描述,主要用来实现操作方法。总线负责将两者配对。
这样可以提高代码的通用性,让驱动可以兼容不同的硬件。至于如何应用,笔者还是不清楚。这里只介绍设备驱动模型,之后会将模型构建出来。
自定义总线 mybus
mybus.c实现device和device的匹配。在模块加载后,注册mybus到总线;每调用一次注册函数(driver_register或者device_register)就会匹配一次,匹配就会执行match函数,match函数执行成功之后,代码会往下走,去执行驱动中的probe函数。
模块加载 --> 注册mybus总线 --> 回调 mybus.match --> 匹配成功
- 构建总线对象
- 注册和注销总线
- 实现回调mybus.match方法
-
//mubus.c
-
#include <linux/init.h>
-
#include <linux/module.h>
-
#include <linux/device.h>
-
-
-
-
int mybus_match(struct device *dev, struct device_driver *drv);
-
-
-
struct bus_type mybus = {//实例化一个bus对象
-
.name = "mybus",
-
.match = mybus_match,
-
};
-
EXPORT_SYMBOL(mybus);
-
-
-
static int __init mybus_init(void)
-
{
-
printk("----------%s-------------\n", __FUNCTION__);
-
-
int ret;
-
ret = bus_register(&mybus); //注册bus
-
if(ret != 0)
-
{
-
printk("bus_register error\n");
-
return ret;
-
}
-
-
-
return ret;
-
}
-
-
static void __exit mybus_exit(void)
-
{
-
printk("----------%s-------------\n", __FUNCTION__);
-
-
//注销bus
-
bus_unregister(&mybus);
-
-
}
-
-
-
int mybus_match(struct device *dev, struct device_driver *drv)
-
{
-
//如果匹配成功,match方法一定要返回一个1, 失败返回0
-
if(!strncmp(drv->name, dev->kobj.name, strlen(drv->name)))
-
{
-
printk("match ok\n");
-
return 1;
-
}else{
-
printk("match failed\n");
-
return 0;
-
-
}
-
return 0;
-
}
-
-
-
-
-
module_init(mybus_init);
-
module_exit(mybus_exit);
-
MODULE_LICENSE("GPL");
注册成功后在系统sys/目录下查看到这个结点
定义设备 mydev
mydev.c用来描述设备信息。将硬件的信息描述成为一个结构体,添加到mydev.platform_data这个属性当中。在模块加载后,注册device到了mybus总线
- 构建device对象
- 将device注册到mybus总线/从mybus总线注销
- 自定义数据类型,构造结构体描述硬件信息
-
//mydev.c
-
#include <linux/init.h>
-
#include <linux/module.h>
-
#include <linux/device.h>
-
#include "dev_info.h"
-
-
-
void mydev_release(struct device *dev);
-
-
-
-
-
extern struct bus_type mybus;
-
-
struct mydev_desc devinfo = {
-
.name = "testdev",
-
.irqno = 9999,
-
.addr = 0x30008000,
-
};
-
-
-
struct device mydev = {
-
.init_name = "fsdev_drv",
-
.bus = &mybus,
-
.release = mydev_release,
-
.platform_data = &devinfo,
-
};
-
-
-
static int __init mydev_init(void)
-
{
-
printk("----------%s-------------\n", __FUNCTION__);
-
//将device注册到总线中
-
int ret;
-
ret = device_register(&mydev);
-
if(ret < 0)
-
{
-
printk("device_register error\n");
-
return ret;
-
}
-
-
return ret;
-
}
-
-
static void __exit mydev_exit(void)
-
{
-
printk("----------%s-------------\n", __FUNCTION__);
-
device_unregister(&mydev);//注销dev
-
-
-
}
-
-
-
void mydev_release(struct device *dev)
-
{
-
-
printk("----------%s-------------\n", __FUNCTION__);
-
}
-
-
-
-
-
-
module_init(mydev_init);
-
module_exit(mydev_exit);
-
MODULE_LICENSE("GPL");
注册成功后在系统sys/目录下查看到这个设备结点
编写驱动 mydrv
mydrv.c用来实现驱动的方法。在模块加载后,注册driver到了mybus总线。当驱动和设备匹配成功后,系统自动调用mydrv.probe这个方法。另外,驱动也通过总线获取设备的信息,"dev_info.h"头文件中声明的是描述设备的结构体类型,是用来接收另一边设备数据的格式。
- 构建driver对象
- 将driver对象注册总线/从总线注销
- 实现驱动的回调mydrv.probe 和 mydrv.remove方法
-
//mydrv.c
-
#include <linux/init.h>
-
#include <linux/module.h>
-
#include <linux/device.h>
-
#include <linux/io.h>
-
#include "dev_info.h"
-
-
-
-
-
int mydrv_probe(struct device *dev);
-
int mydrv_remove(struct device *dev);
-
-
-
-
extern struct bus_type mybus;
-
-
struct device_driver mydrv = {
-
.name = "fsdev_drv",
-
.bus = &mybus,
-
.probe = mydrv_probe,
-
.remove = mydrv_remove,
-
-
};
-
-
-
static int __init mydrv_init(void)
-
{
-
printk("----------%s-------------\n", __FUNCTION__);
-
//将driver注册到总线中
-
int ret;
-
ret = driver_register(&mydrv);
-
if(ret < 0)
-
{
-
printk("device_register error\n");
-
return ret;
-
}
-
-
return ret;
-
}
-
-
static void __exit mydrv_exit(void)
-
{
-
printk("----------%s-------------\n", __FUNCTION__);
-
driver_unregister(&mydrv);
-
-
}
-
-
-
struct mydev_desc *pdesc;
-
int mydrv_probe(struct device *dev)
-
{
-
printk("----------%s-------------\n", __FUNCTION__);
-
-
pdesc = (struct mydev_desc *)dev->platform_data;
-
-
printk("name = %s\n", pdesc->name);
-
printk("irqno = %d\n", pdesc->irqno);
-
-
-
return 0;
-
-
}
-
-
int mydrv_remove(struct device *dev)
-
{
-
-
printk("----------%s-------------\n", __FUNCTION__);
-
return 0;
-
}
-
-
-
-
-
-
module_init(mydrv_init);
-
module_exit(mydrv_exit);
-
MODULE_LICENSE("GPL");
注册成功后在系统sys/目录下查看到这个结点
文章来源: blog.csdn.net,作者:hinzer,版权归原作者所有,如需转载,请联系作者。
原文链接:blog.csdn.net/feit2417/article/details/84308266
- 点赞
- 收藏
- 关注作者
评论(0)