Linux驱动 device 的probe函数是怎么被调用的

今天正好有空,研究了一下platformdevice的probe函数时如何被调用的。我觉得这个过程应该可以推广到一般设备的探测函数的调用。

以mini2440中的watchdog为例。

先看配置文件中对watchdog的设置:

static struct resource s3c_wdt_resource[] = {
	[0] = {
		.start = S3C24XX_PA_WATCHDOG,
		.end   = S3C24XX_PA_WATCHDOG + S3C24XX_SZ_WATCHDOG - 1,
		.flags = IORESOURCE_MEM,
	},
	[1] = {
		.start = IRQ_WDT,
		.end   = IRQ_WDT,
		.flags = IORESOURCE_IRQ,
	}

};

struct platform_device s3c_device_wdt = {
	.name		  = "s3c2410-wdt",
	.id		  = -1,
	.num_resources	  = ARRAY_SIZE(s3c_wdt_resource),
	.resource	  = s3c_wdt_resource,
};


 

在系统启动时 mini2440_machine_init调用了 platform_add_devices(mini2440_devices, ARRAY_SIZE(mini2440_devices)); 其中 mini2440_devices 中就包括了s3c_deice_wdt。

int platform_add_devices(struct platform_device **devs, int num)
{
	int i, ret = 0;

	for (i = 0; i < num; i++) {
		ret = platform_device_register(devs[i]);
		if (ret) {
			while (--i >= 0)
				platform_device_unregister(devs[i]);
			break;
		}
	}

	return ret;
}
int platform_device_register(struct platform_device *pdev)
{
	device_initialize(&pdev->dev);
	return platform_device_add(pdev);
}
int platform_device_add(struct platform_device *pdev)
{
	int i, ret = 0;

	if (!pdev)
		return -EINVAL;

	if (!pdev->dev.parent)
		pdev->dev.parent = &platform_bus;

	pdev->dev.bus = &platform_bus_type;

	if (pdev->id != -1)
		dev_set_name(&pdev->dev, "%s.%d", pdev->name,  pdev->id);
	else
		dev_set_name(&pdev->dev, "%s", pdev->name);

	for (i = 0; i < pdev->num_resources; i++) {
		struct resource *p, *r = &pdev->resource[i];

		if (r->name == NULL)
			r->name = dev_name(&pdev->dev);

		p = r->parent;
		if (!p) {
			if (resource_type(r) == IORESOURCE_MEM)
				p = &iomem_resource;
			else if (resource_type(r) == IORESOURCE_IO)
				p = &iopor
  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
adc_keys_probe函数Linux内核中与ADC按键相关的设备树探测函数。在设备树中,如果有ADC按键的相关信息(如所使用的ADC控制器、引脚等),Linux内核会自动调用函数进行探测。 其主要功能包括: 1. 读取设备树中ADC按键节点的相关信息,如所用ADC控制器和引脚号; 2. 根据以上信息初始化ADC控制器,并将其与对应的GPIO引脚进行绑定; 3. 注册Linux输入子系统的按键输入设备,并将其与初始化好的ADC控制器进行关联。 下面是该函数的代码实现: static int adc_keys_probe(struct platform_device *pdev) { struct device *dev = &pdev->dev; struct adc_keys_platform_data *pdata = dev_get_platdata(dev); const char *adc_name = pdata ? pdata->adc_name : NULL; struct input_dev *input_dev; struct adc_keys *keys; int ret, i; if (!adc_name) { dev_err(dev, "no ADC controller specified in platform data\n"); return -EINVAL; } input_dev = devm_input_allocate_device(dev); if (!input_dev) return -ENOMEM; keys = devm_kzalloc(dev, sizeof(*keys), GFP_KERNEL); if (!keys) return -ENOMEM; platform_set_drvdata(pdev, keys); keys->input = input_dev; keys->adc = devm_iio_channel_get(&pdev->dev, "iio"); if (IS_ERR(keys->adc)) { dev_err(dev, "failed to get ADC channel\n"); ret = PTR_ERR(keys->adc); goto err_free_mem; } input_dev->name = pdev->name; input_dev->phys = "keys/input0"; input_dev->id.bustype = BUS_HOST; input_dev->id.vendor = 0x0001; input_dev->id.product = 0x0001; input_dev->id.version = 0x0100; input_set_capability(input_dev, EV_KEY, KEY_POWER); input_set_capability(input_dev, EV_KEY, KEY_VOLUMEUP); input_set_capability(input_dev, EV_KEY, KEY_VOLUMEDOWN); keys->min_val = pdata ? pdata->min_val : ADC_KEYS_DEFAULT_MAX; keys->max_val = pdata ? pdata->max_val : ADC_KEYS_DEFAULT_MIN; ret = input_register_device(input_dev); if (ret) { dev_err(dev, "failed to register input device\n"); goto err_free_mem; } ret = adc_keys_init_dev(keys, adc_name); if (ret) { dev_err(dev, "failed to init ADC controller\n"); goto err_free_dev; } for (i = 0; i < ARRAY_SIZE(keys->keymap); i++) { ret = input_register_keycode(input_dev, keys->keymap[i].type, keys->keymap[i].code, NULL); if (ret) { dev_err(dev, "failed to register input keycode\n"); goto err_free_dev; } } ret = adc_keys_set_timer_interval(keys); if (ret) dev_warn(dev, "Failed to initialize the polling timer\n"); dev_info(dev, "registered ADC keys input device\n"); return 0; err_free_dev: input_unregister_device(input_dev); err_free_mem: return ret; } 该函数通过dev_get_platdata函数读取设备树节点的平台数据信息,获取相关参数。接下来,分别进行输入子系统的相关初始化、iio_channel获取、ADC控制器的初始化、按键注册及关联操作,最终成功时输出相关信息并返回0,失败则进行相应的错误处理操作。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值