前置:这里使用的linux版本是4.8,x86体系。

其实linux的内核启动的入口文件还是非常好找的,init/main.c。

static 和 extern

首先理解的是static和extern的区别:

static int kernel_init(void *);

extern void init_IRQ(void);
extern void fork_init(void);
extern void radix_tree_init(void);

这个代码说的是kernel_init函数的定义在这个文件中,extern说明init_IRQ函数的定义在其他文件中。

这三个extern分别是对中断的初始化,对fork功能的初始化,对基数树的初始化。不过具体不知道为什么有的函数以init_xxx为风格,有的又以xxx_init的风格来做。

main的第一行看到了这么个语句

#define DEBUG

感觉有点奇怪,原来还有#define <宏名> 而没有定义具体的值。其实这个可以当作已经有定义,且定义了空串来理解。

http://bbs.csdn.net/topics/390960776?page=1

继续往下面看,还会看到

bool early_boot_irqs_disabled __read_mostly;

这里最后的__read_mostly 是一个宏,它标记了前面这个变量是很经常被读取的。那么做了标记有什么用呢?

如果在有缓存的平台上,它就能把这个变量存放到cache中,以保证后续读取的速度。这个宏定义在 arch/arm/include/asm/cache.h

#define __read_mostly __attribute__((__section__(".data..read_mostly")))

这里的意思是将这个数据结构链接进data.read_mostly段。

EXPORT_SYMBOL

EXPORT_SYMBOL(system_state);

这个是和extern一起使用的,表示system_state这个方法在这个模块中定义了,提供给其他模块使用。

在其他模块中,只需要使用extern 就可以使用这个方法。

这里就涉及到模块的概念。

模块是linux内核对外提供的一个插件机制,由于linux是单内核,这个单内核是相对微内核来说的。所以linux很大可能会非常庞大,这个模块机制就是对单内核的一种补充,把一些功能放给内核模块开发。比如像上面的那个代码,就是对内核提供了system_state的函数接口。

__initdata

下面代码:

char __initdata boot_command_line[COMMAND_LINE_SIZE];

这里的__initdata也是一个宏,定义在include/linux/init.h

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

同上面__read_mostly一样,是用来把这个变量绑定在某个区里面。

http://blog.csdn.net/beatbean/article/details/8448623

这个图就说明了什么是__section。它的功能有点像是全局变量,只是这个全局变量是对汇编这个层次的表达,某个变量,我固定在某个内存段里面。这么做其实还有一个好处,段也是一种分类,比如这个段存储的是init函数的变量,那么等初始化结束之后,我把这个段的内存直接释放。里面的变量也一次性消除了。

EXPORT_SYMBOL_GPL

下面看到一个很奇怪的方法

EXPORT_SYMBOL_GPL(static_key_initialized);

这个和之前的EXPORT_SYMBOL不一样,多了一个GPL后缀。

由于模块很有可能是第三方(非linux内核组成员)开发的。那么有人希望自己开发的模块是闭源的。它就会在自己开发的模块里面使用

MODULE_LICENSE("Proprietary")

来标记这个模块是闭源的。相对的,如果你的模块遵循GPL这个开源许可证规则,那么则增加下面的:

MODULE_LICENSE("GPL");

好了,linux对这两种许可证行为的模块开放的接口并不相同,本节的这个函数就是说明这个方法只对GPL的模块开放。

http://www.ruanyifeng.com/blog/2010/02/why_gpl_is_a_better_choice.html

__setup 和 early_param

unsigned int reset_devices;

EXPORT_SYMBOL(reset_devices);

static int __init set_reset_devices(char *str)

{

reset_devices = 1;

return 1;

}

__setup("reset_devices", set_reset_devices);

这段代码,首先需要理解__setup,这个函数就理解为:启动时候如果有接收reset_devices参数,那么就调用set_reset_devices方法。而详细看了下set_reset_devices方法,里面只是把reset_devices变量设置为1,但是呢,这个reset_devices变量又是一个给所有模块使用的变量。

所以这段代码能达到的功能是只要启动参数有包含reset_device,通过设置reset_devices通知给所有模块。

与__setup相对应的还有一个叫做early_param。这两个宏函数的功能一样,区别就在于early_param定义的参数比__setup更早。

http://www.linuxde.net/2013/02/12446.html

linux源码分析(一)的更多相关文章

  1. linux源码分析2

    linux源码分析 这里使用的linux版本是4.8,x86体系. 这篇是 http://home.ustc.edu.cn/~boj/courses/linux_kernel/1_boot.html  ...

  2. Linux源码分析之:malloc、free

    之前写代码的时候一直有个疑问,malloc申请内存的时候指定了内存大小,但是free的时候却只指定要释放的内存地址,那么free是如何知道它要释放的内存空间大小呢? 源码之前,了无秘密,下面就从源码来 ...

  3. linux源码分析(五)-start_kernel

    前置:这里使用的linux版本是4.8,x86体系. local_irq_disable(); 这个函数是做了关闭中断操作.和后面的local_irq_enable相对应.说明启动的下面函数是不允许被 ...

  4. linux源码分析(四)-start_kernel-cgroup

    前置:这里使用的linux版本是4.8,x86体系. cgroup_init_early(); 聊这个函数就需要先了解cgroup. cgroup概念 这个函数就是初始化cgroup所需要的参数的.c ...

  5. linux源码分析(三)-start_kernel

    前置:这里使用的linux版本是4.8,x86体系. start_kernel是过了引导阶段,进入到了内核启动阶段的入口.函数在init/main.c中. set_task_stack_end_mag ...

  6. linux源码分析(二)-启动过程

    前置:这里使用的linux版本是4.8,x86体系. 这篇是 http://home.ustc.edu.cn/~boj/courses/linux_kernel/1_boot.html 的学习笔记. ...

  7. linux源码分析 - 进程

    本文为原创,转载请注明:http://www.cnblogs.com/tolimit/ 最近在回想一些知识点的时候,觉得对进程这一块有些模糊,特别写一篇随笔对进程信息进行巩固和复习. 程序和进程 以我 ...

  8. 源码分析:动态分析 Linux 内核函数调用关系

    源码分析:动态分析 Linux 内核函数调用关系 时间 2015-04-22 23:56:07  泰晓科技 原文  http://www.tinylab.org/source-code-analysi ...

  9. Linux内核源码分析方法

    一.内核源码之我见 Linux内核代码的庞大令不少人“望而生畏”,也正因为如此,使得人们对Linux的了解仅处于泛泛的层次.如果想透析Linux,深入操作系统的本质,阅读内核源码是最有效的途径.我们都 ...

随机推荐

  1. Android零散

    2016-03-13 Android零散 ListView中嵌套GridView 要实现分组列表这样的效果:点击ListView中的分组名称,即展开此分组显示其包含的项目.使用ExpandableLi ...

  2. .NET面试题系列[14] - LINQ to SQL与IQueryable

    .NET面试题系列目录 名言警句 "理解IQueryable的最简单方式就是,把它看作一个查询,在执行的时候,将会生成结果序列." - Jon Skeet LINQ to Obje ...

  3. 一步步学习javascript基础篇(6):函数表达式之【闭包】

    回顾前面介绍过的三种定义函数方式 1. function sum (num1, num2) { return num1 + num2; }  //函数声明语法定义 2. var sum = funct ...

  4. Hadoop学习笔记—15.HBase框架学习(基础实践篇)

    一.HBase的安装配置 1.1 伪分布模式安装 伪分布模式安装即在一台计算机上部署HBase的各个角色,HMaster.HRegionServer以及ZooKeeper都在一台计算机上来模拟. 首先 ...

  5. .Net开发笔记(二十一) 反射在.net中的应用

    反射概念在网上到处都有,但是讲到的具体的应用很少,一个重要的原因是现实中真的很少用得到它.引用msdn上对“反射”的解释: "通过 System.Reflection 命名空间中的类以及 S ...

  6. iOS开发系列--C语言之指针

    概览 指针是C语言的精髓,但是很多初学者往往对于指针的概念并不深刻,以至于学完之后随着时间的推移越来越模糊,感觉指针难以掌握,本文通过简单的例子试图将指针解释清楚,今天的重点有几个方面: 什么是指针 ...

  7. 【Java并发编程实战】-----“J.U.C”:Condition

    在看Condition之前,我们先来看下面这个例子: 工厂类,用来存放.取出商品: public class Depot { private int depotSize; //仓库大小 private ...

  8. 说说SQL Server 网络配置

    打开Sql Server Configuration Manager,里面显示了SQL Server的网络配置,这些到底表示什么含义呢? 图一:MSSQLSERVER的协议 这些配置选项,其实就是为了 ...

  9. php相对于java、js几点不太一样的地方

    1.PHP 变量作用域 在 PHP 中,可以在脚本的任意位置对变量进行声明. 变量的作用域指的是变量能够被引用/使用的那部分脚本. PHP 有三种不同的变量作用域: local(局部) global( ...

  10. 无法在“EntityFramework”已存在的情况下创建影像复制该文件的解决方案

    问题产生的原因:你项目正在生成中你就打开浏览器预览了,导致这个问题解决方案:右击重新生成项目,等生成后再打开 “/”应用程序中的服务器错误. 无法在“EntityFramework”已存在的情况下创建 ...