Linux源码__init宏

阅读数:118 评论数:0

跳转到新版页面

分类

Linux

正文

一、概述

在 Linux 内核开发中,__init 宏是一个特殊的宏,用于标记某些函数或数据是初始化代码或数据。当内核启动并完成初始化后,这部分被标记为 __init 的代码和数据可以被丢弃,以节省内存。这对于嵌入式系统或资源有限的环境特别有用,因为它允许系统在不再需要初始化代码后释放出这部分内存供其他用途使用。

二、源码相关

在内核里经常可以看到__init, __devinit这样的语句,这都是在init.h中定义的宏, gcc在编译时会把被修饰的内容放到这些宏所代表的section中。

#define __init __section(.init.text) __cold __latent_entropy __noinitretpoline
#define __initdata __section(.init.data)
#define __initconst __section(.init.rodata)
#define __exitdata __section(.exit.data)
#define __exit_call __used __section(.exitcall.exit)

#define __section(section)              __attribute__((__section__(section)))

__section(.init.text)指放到.init.text的数据段中。这样系统在初始化,只需要把.init.text数据段中所有初始化执行一遍即可,当系统启动完毕后,这个段中的内存就会被释放供其它使用。

#define __cold                          __attribute__((__cold__))

__cold标识该函数针对大小而不是速度进行优化。

【include/linux/complier-gcc.h】
#if defined(LATENT_ENTROPY_PLUGIN) && !defined(__CHECKER__)
#define __latent_entropy __attribute__((latent_entropy))
#endif

__latent_entropy用于函数,那么plugin将对其进行检测,用于变量,那么plugin将用一个随机值初始化它。

【include/linux/complier-gcc.h】
#ifdef CONFIG_RETPOLINE
#define __noretpoline __attribute__((__indirect_branch__("keep")))
#endif

/* Built-in __init functions needn't be compiled with retpoline */
#if defined(__noretpoline) && !defined(MODULE)
#define __noinitretpoline __noretpoline
#else
#define __noinitretpoline    /一般情况下,定义了MODULE,就会走这个分支**/
#endif

 

其典型用法如下:

static int __init xxx_drv_init(void)
{
return pci_register_driver(&xxx_driver);
}

1、一部分内核初始化机制依赖它。

如kernel将初始化要执行的init函数,分为7个级别,core_initcall, postcore_initcall, arch_initcall, subsys_initcall, fs_initcall, device_initcall, late_initcall。这7个级别优行列级递减,即先执行core_initcall,最后执行late_initcall。通过使用文中提到的宏, gcc会将初始化代码按下面结构安排。

在内核初始化时,从__initcall_start到__initcall_end之间的initcall被执行一次。

2、提高系统效率

初始化代码的特点是:在系统启动时运行,且一旦运行马上退出内存,不再占用内存。




相关推荐

【include/linux/init.h】 #define __init __section(".init.text") __cold __latent_entropy __noinitretpo

一、概述 1、为什么Linus不使用GPLv3 在 PC 上,只要你得到了某个程序的源代码,就可以自行编译生成二进制程序,然后替换掉原有的二进制程序,你的程序自由很容易得到保证。然而 iPod、iPh

说明:这种方式只是用于方便阅读代码,因为可以在源代间快速索引跳跃。但是最后可能会有一些warning,可以不必关心,如果是强迫症,可以使用下面这种方式来去掉。

一、概述 在Linux系统中,/usr/bin和/usr/local/bin是两个常见的目录,用于存放可执行文件(二进制文件)。 很多应用都安装在/usr/local下面,先看一下automake工具

  一、概述 vmstat命令是最常见的Linux/Unix监控工具,可以监控给定时间间隔服务器的CPU使用率、内存使用、IO情况。相比top命令,可以查看到整个机器的CPU、内存、IO的使用情况,而

一、概述 sar,System Activity Reporter。是目前 Linux 上最为全面的系统性能分析工具之一,可以从多方面对系统的活动进行报告,包括:文件的读写情况、系统调用的使用情况、磁

一、简介 简单来说awk就是把文件逐行的读入,以空格为默认分隔符将每行切片,切开的部分再进行各种分析处理。 二、使用方法 1、基本语法  awk '条件类型1 {动作1} 条件类型2{动作2} ...

一、概述 列出目标目录中所有的子目录和文件。 二、 语法 ls [选项] [目录名] -a, –all 列出目录下的所有文件,包括以 . 开头的隐含文件 -A 同-a,但不列出“.”(表示当前目录)

一、概述 cd全称是change directory,用于切换当前工作目录。 注意的是,cd命令是一个内建命令,它是由 shell 提供的。因此,不同的 shell 可能会有一些差异,但基本的用法和功

一、概述 全称为print working directory,查看”当前工作目录“的完整路径,一般情况下不带任何参数 二、语法 pwd [选项] -L 即logical,逻辑路径 -P 即