__define_initcall(level,fn)

在linux的驱动里经常会出现__define_initcall(level,fn)的用法,特别是在驱动,的init函数中,譬如I2C_ADAPTER的驱动有这样的函数subsys_initcall(imapx200_i2c_adap_init);关于它的具体用法,就参考下面的资料

 

宏定义__define_initcall(level,fn)对于内核的初始化很重要,他指示编译器在编译的时候,将一系列初始化函数的起始地址值按照一定的顺序放在一个section中。在内核初始化段,do_initcalls() 将按顺序从该section中以函数指针的形式取出这些函数的起始地址,来依次完成相应的初始化。于内核某些部分的初始化需要依赖于其他某些部分的初始化的完成,因此这个顺序排列常常很重要。 

  下面将从__define_initcall(level,fn) 宏定义的代码分析入手,依次分析名称为initcall.init的section的结构,最后分析内核初始化函数do_initcalls()是怎样利用宏定义__define_initcall(level,fn)及其相关的衍生的7个宏宏定义,来实现内核某些部分的顺序初始化的。

1、分析 __define_initcall(level,fn) 宏定义
1) 这个宏的定义位于inlclude\linux\init.h中:
#define __define_initcall(level,fn)   \
static initcall_t __initcall_##fn  \
__attribute__((__section__(".initcall" level ".init"))) \
= fn
其中 initcall_t 是个函数指针类型:
typedef int (*initcall_t)(void);
而属性 __attribute__((__section__())) 则表示把对象放在一个这个由括号中的名称所指代的section中。所以这个宏定义的的含义是:1) 声明一个名称为__initcall_##fn的函数指针(其中##表示替换连接,);2) 将这个函数指针初始化为fn;3) 编译的时候需要把这个函数指针变量放置到名称为 ".initcall" level ".init"的section中(比如level="1",代表这个section的名称是 ".initcall1.init")。
2) 举例:__define_initcall(6, pci_init)
上述宏调用的含义是:1) 声明一个函数指针__initcall_pic_init = pci_init;且 2) 这个指针变量__initcall_pic_init 需要放置到名称为 .initcall6.init的section中( 其实质就是将 这个函数pic_init的首地址放置到了这个section中)。
3) 这个宏一般并不直接使用,而是被定义成下述其他更简单的7个衍生宏
这些衍生宏宏的定义也位于 inlclude\linux\Init.h 中:
#define core_initcall(fn)         __define_initcall("1",fn)
#define postcore_initcall(fn)     __define_initcall("2",fn)
#define arch_initcall(fn)         __define_initcall("3",fn)
#define subsys_initcall(fn)       __define_initcall("4",fn)
#define fs_initcall(fn)           __define_initcall("5",fn)
#define device_initcall(fn)       __define_initcall("6",fn)
#define late_initcall(fn)         __define_initcall("7",fn)
因此通过宏 core_initcall() 来声明的函数指针,将放置到名称为.initcall1.init的section中,而通过宏 postcore_initcall() 来声明的函数指针,将放置到名称为.initcall2.init的section中,依次类推。
4) 举例:device_initcall(pci_init)
解释同上 1-2)。
2、和初始化调用有关section--initcall.init被分成了7个子section
1) 他们依次是.initcall1.init、.initcall2.init、...、.initcall7.init
2) 按照先后顺序依次排列
3) 他们的定义在文档vmlinux.lds.S中
例如 对于i386+,在i386\kernel\vmlinux.lds.S中有:
__initcall_start = .;
.initcall.init : {
*(.initcall1.init)
*(.initcall2.init)
*(.initcall3.init)
*(.initcall4.init)
*(.initcall5.init)
*(.initcall6.init)
*(.initcall7.init)
}
__initcall_end = .;
而在makefile 中有

LDFLAGS_vmlinux += -T arch/$(ARCH)/kernel/vmlinux.lds.s
4) 在这7个section总的开始位置被标识为__initcall_start,而在结尾被标识为__initcall_end。
3、 内核初始化函数do_basic_setup(): do_initcalls() 将从.initcall.init中,也就是这7个section中依次取出任何的函数指针,并调用这些函数指针所指向的函数,来完成内核的一些相关的初始化。
这个函数的定义位于init\main.c中:
extern initcall_t __initcall_start, __initcall_end;
static void __init do_initcalls(void)
{
initcall_t *call;
....
for (call = &__initcall_start; call  

***********************************************************************

假如您希望某个初始化函数在内核初始化阶段就被调用,那么您就应该使用宏__define_initcall(level,fn) 或 其7个衍生宏来把这个初始化函数fn的起始地址按照初始化的顺序放置到相关的section 中。 内核初始化时的do_initcalls()将从这个section中按顺序找到这些函数来执行。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
下列代码在输入能耗之后回车会直接输出最终结果无法输入CCM值,请改良下列代码。#include <stdio.h> // 定义各项标准的等级划分 #define CADR_LEVEL_1 350 #define CADR_LEVEL_2 450 #define CADR_LEVEL_3 600 #define NOISE_LEVEL_1 30 #define NOISE_LEVEL_2 40 #define NOISE_LEVEL_3 50 #define POWER_CONSUMPTION_LEVEL_1 0.025 #define POWER_CONSUMPTION_LEVEL_2 0.04 #define POWER_CONSUMPTION_LEVEL_3 0.07 #define CCM_LEVEL_1 10 #define CCM_LEVEL_2 13 #define CCM_LEVEL_3 16 // 定义评价函数 int evaluate(int cadr, int noise, int power_consumption, int ccm) { int cadr_level = 0, noise_level = 0, power_consumption_level = 0, ccm_level = 0; // 判断 CADR 级别 if (cadr >= CADR_LEVEL_3) { cadr_level = 3; } else if (cadr >= CADR_LEVEL_2) { cadr_level = 2; } else if (cadr >= CADR_LEVEL_1) { cadr_level = 1; } // 判断噪音级别 if (noise <= NOISE_LEVEL_1) { noise_level = 3; } else if (noise <= NOISE_LEVEL_2) { noise_level = 2; } else if (noise <= NOISE_LEVEL_3) { noise_level = 1; } // 判断能耗级别 if (power_consumption <= POWER_CONSUMPTION_LEVEL_1) { power_consumption_level = 3; } else if (power_consumption <= POWER_CONSUMPTION_LEVEL_2) { power_consumption_level = 2; } else if (power_consumption <= POWER_CONSUMPTION_LEVEL_3) { power_consumption_level = 1; } // 判断 CCM 级别 if (ccm >= CCM_LEVEL_3) { ccm_level = 3; } else if (ccm >= CCM_LEVEL_2) { ccm_level = 2; } else if (ccm >= CCM_LEVEL_1) { ccm_level = 1; } // 计算综合评价 int total_evaluation = cadr_level+noise_level + power_consumption_level + ccm_level; return total_evaluation; } int main() { int cadr,noise, power_consumption, ccm; // 读取各项参数值 printf("请输入 CADR 值:"); scanf("%d", &cadr); printf("请输入噪音值:"); scanf("%d", &noise); printf("请输入能耗值:"); scanf("%d", &power_consumption); printf("请输入 CCM 值:"); scanf("%d", &ccm); // 计算评价结果 int evaluation = evaluate(cadr,noise, power_consumption, ccm); printf("总评价等级为:%d\n", evaluation); return 0; }
05-10

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zyhorse2010

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值