linux3.* 内核的__setup函数

在内核代码中经常会看到__setup函数,这个函数从哪来,到哪去?下面我们逐一分析

例如在I2C驱动中就有这么一个setup函数:

__setup("i2c_bus=", omap_i2c_bus_setup);

我们先要直到它是从哪来的:看如下代码:

#define __setup_param(str, unique_id, fn, early)			\
	static const char __setup_str_##unique_id[] __initconst	\
		__aligned(1) = str; \
	static struct obs_kernel_param __setup_##unique_id	\
		__used __section(.init.setup)			\
		__attribute__((aligned((sizeof(long)))))	\
		= { __setup_str_##unique_id, fn, early }

#define __setup(str, fn)					\
	__setup_param(str, fn, fn, 0)

可以清除的看出来__setup函数最后被宏替代成__setup_param函数,去掉纷繁复杂的各种连接。我们利用__setup("reset_devices", set_reset_devices);看看最终的表达形式是什么

static const char __setup_str_omap_i2c_bus_setup[]= "i2c_bus=";
static struct obs_kernel_param __setup_omap_i2c_bus_setup = { __setup_str_omap_i2c_bus_setup, omap_i2c_bus_setup, 0}

而最终的函数实现体现在初始化函数中如下:

static int __init obsolete_checksetup(char *line)
{
	const struct obs_kernel_param *p;
	int had_early_param = 0;

	p = __setup_start;
	do {
		int n = strlen(p->str);
		if (parameqn(line, p->str, n)) {
			if (p->early) {
				/* Already done in parse_early_param?
				 * (Needs exact match on param part).
				 * Keep iterating, as we can have early
				 * params and __setups of same names 8( */
				if (line[n] == '\0' || line[n] == '=')
					had_early_param = 1;
			} else if (!p->setup_func) {
				printk(KERN_WARNING "Parameter %s is obsolete,"
				       " ignored\n", p->str);
				return 1;
			} else if (p->setup_func(line + n))
				return 1;
		}
		p++;
	} while (p < __setup_end);

	return had_early_param;
}

这里的意思是从__setup_start开始处到__setup_end处中查找一个数据结构,这个数据结构中有str与setup_func这两个数据成员变量. 只要与这里面的str与输入的参数字符串相匹配,就会调用个这个字符串后面所指的内容.意思也就是调用
omap_i2c_bus_setup
函数,而这个函数应该在你的代码中体现出来的。自己的初始化函数。





你可能感兴趣的:(linux随笔)