dev_set_drvdata, dev_get_drvdata machanism

example:
static int vrha_fpga_dtb_probe(struct vrha_device *vdev)
//tianxianting, vrha_device* already created by function "vrha_device_create", so if matched
//pass vrha_device* to this probe
{
struct device_node *np = vdev->dev.of_node;
struct device_node *epcq_np;
struct device *dev = &vdev->dev;
struct fpga_dtb_dev *fdd;
const unsigned int *prop_dt;
uint32_t prop_img[2];
int err, i;
/* Allocate memory for the fpga_dtb_device structure */
fdd = devm_kzalloc(dev, sizeof(*fdd), GFP_KERNEL);
...
err = sysfs_create_group(&dev->kobj, &vrha_fpga_dtb_attr_group);
...
dev_set_drvdata(dev, fdd);
...
}
Use it in this device's attr file:
/* sysfs entries for reading FPGA image stored versions */
static DEVICE_ATTR(app1_version, S_IRUGO,
vrha_fpga_app1_img_show_version, NULL);
static ssize_t vrha_fpga_app1_img_show_version(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct fpga_dtb_dev * fdd= dev_get_drvdata(dev);
return vrha_fpga_dt_get_stored_version(fdd, FPGA_DTB_APP1, "version", buf);
}

在学习利用sysfs,实现对设备的驱动过程中,
void *dev_get_drvdata(const struct device *dev)函数,看内核device.h中的注释,说是获取驱动数据指针。
但是不知道怎么用。能否有人给个例子
真心感谢~
例子在内核代码里面一大堆,首先要明白这个data指的是驱动的私有数据,也就是说不一样的驱动私有数据是不一样的,结构体也不一样, 这就是为什么要用void类型的指针。
多看点代码吧,内核是最好的老师。
--- 共有 1 条评论 ---
EasyL: 谢谢~ 嗯,看了点代码,了解您说的不一样的driver数据不一样这一点。 看到的代码里都是直接使用此函数取driver的数据。 疑惑在,没有看到什么时候写的数据,就能把自己定义的结构体类型的数据取出来吗?
在Linux Driver的代码中,我们经常看到标题中的函数。那么这个函数究竟如何使用,它的工作原理又是什么?下面,我们来详细看一下这个函数的实现。
driver/base/dd.c 
/*
 * These exports can't be _GPL due to .h files using this within them, and it
 * might break something that was previously working...
 */
void *dev_get_drvdata(const struct device *dev)
{
	if (dev && dev->p)
		return dev->p->driver_data;
	return NULL;
}
EXPORT_SYMBOL(dev_get_drvdata);

int dev_set_drvdata(struct device *dev, void *data)
{
	int error;

	if (!dev->p) {
		error = device_private_init(dev);
		if (error)
			return error;
	}
	dev->p->driver_data = data;
	return 0;
}
EXPORT_SYMBOL(dev_set_drvdata);
从第4行代码中,我们可以看到此函数主要是返回了device->p->driver_data指针。那么,我们下面来看一下, Kernel中比较重要的Device结构体,它其实是对内核中所有设备的抽象表示。 所有的设备都有一个device实例与之对应,而且Device结构体的主要用法为将其嵌入到其他的设备结构体中,如platform_device等。同时,Device结构体也负责作为子系统之间交互的统一参数。那么,我们下面主要看一下,device结构体的构成。
include/linux/device.h
struct device {
     struct device *parent;
     struct device_private *p;   //负责保存driver核心部分的数据
     struct kobject kobj;
     const char *init_name;
     .......
     struct device_driver *driver;
#ifdef CONFIG_PINCTRL
     struct dev_pin_info *pins;
#endfi
     .......
     struct device_node *of_node;  //负责保存device_tree中相应的node地址
     .......
     const struct attribute_group **groups;
     ......
}
struct device_private {
     struct klist klist_children;
     struct klist_node knode_parent;
     struct klist_node knode_driver;
     struct klist_node knode_bus;
     struct list_head deferred_probe;
     void *driver_data;      //负责保存driver中相应的driver_data
     struct device *device;
}
那么,driver_data是何时进行初始化的呢?我们通过追踪代码是可以发现,一般driver_data的初始化是发生在Driver文件中的probe函数中的。
在probe函数中,malloc完相应的driver data结构体,填充完相应的域后,就会将driver data的地址赋值给driver data。这样,在实现与其他子系统交互的接口时,就能通过其他子系统传递过来的device指针来找到相应的driver data。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值