linux内核分析第三周作业

实验截图

start_kernel代码分析

asmlinkage __visible void __init start_kernel(void)
{
    char *command_line; //地址指针,指向内核启动参数在内存中的位置
    char *after_dashes;

    /*
     * Need to run as early as possible, to initialize the
     * lockdep hash:
     */
    lockdep_init();                      //内核调试模块,用于检查内核互斥机制潜在的死锁问题。
    set_task_stack_end_magic(&init_task);  //init_task为0号进程的PCB
    smp_setup_processor_id();   //查看是否为多处理器平台,获取当前处理器逻辑号,如果是单CPU这个函数什么也不会做。
    debug_objects_early_init();//对调试对象进行早期的初始化,其实就是HASH锁和静态对象池进行初始化 

    /*
     * Set up the the initial canary ASAP:
     */
    boot_init_stack_canary(); //初始化防止栈溢攻击保护的堆栈

    cgroup_init_early(); //初始化进程及子进程性能控制机制

    local_irq_disable();//关闭系统总中断
    early_boot_irqs_disabled = true;

/*
 * Interrupts are still disabled. Do necessary setups, then
 * enable them
 */
    boot_cpu_init();    //激活当前CPU
    page_address_init();  //与高端内存有关未定义为空函数
    pr_notice("%s", linux_banner); //打印内核版本信息,内核启动第一行信息来自这
    setup_arch(&command_line); //内核架构相关初始化函数
    mm_init_cpumask(&init_mm);  //初始化cpu屏蔽字
    setup_command_line(command_line);//对command_line进行备份和保存
    setup_nr_cpu_ids();   //针对SMP处理器的内存初始化函数,如果不是SMP系统都为空函数
    setup_per_cpu_areas(); //若为SMP多核处理器,则给每个cpu分配内存,并拷贝.data.percpu段的数据。为系统中的每个CPU的per_cpu变量申请空间并为boot CPU设置一些数据。 
    smp_prepare_boot_cpu(); /* arch-specific boot-cpu hooks */

    build_all_zonelists(NULL, NULL);// 建立系统内存页区(zone)链表 
    page_alloc_init();//内存页初始化 

    pr_notice("Kernel command line: %s\n", boot_command_line);
    parse_early_param();//解析早期的内核参数
    /*函数对Linux启动命令行参数进行在分析和处理, 
当不能够识别前面的命令时,所调用的函数。*/ 
    after_dashes = parse_args("Booting kernel",
                  static_command_line, __start___param,
                  __stop___param - __start___param,
                  -1, -1, &unknown_bootoption);
    if (!IS_ERR_OR_NULL(after_dashes))
        parse_args("Setting init args", after_dashes, NULL, 0, -1, -1,
               set_init_arg);

    jump_label_init();

    /*
     * These use large bootmem allocations and must precede
     * kmem_cache_init()
     */
    setup_log_buf(0);//使用bootmem分配一个启动信息的缓冲区
    pidhash_init();//使用bootmem分配并初始化PID散列表
    vfs_caches_init_early();//前期VFS缓存初始化
    sort_main_extable();/对内核异常表进行排序
    trap_init(); //初始化中断向量
    mm_init();  //内存管理模块初始化

    /*
     * Set up the scheduler prior starting any interrupts (such as the
     * timer interrupt). Full topology setup happens at smp_init()
     * time - but meanwhile we still have a functioning scheduler.
     */
    sched_init();//调度模块初始化
    /*
     * Disable preemption - early bootup scheduling is extremely
     * fragile until we cpu_idle() for the first time.
     */
    preempt_disable();//禁用抢占和中断,早期启动时期,调度是极其脆弱的
    if (WARN(!irqs_disabled(),
         "Interrupts were enabled *very* early, fixing it\n"))
        local_irq_disable();
    idr_init_cache();//为IDR机制分配缓存
    rcu_init();//内核RCU机制初始化
    context_tracking_init();
    radix_tree_init();//内核radix树算法初始化
    /* init some links before init_ISA_irqs() */
    early_irq_init();//前期外部中断描述符初始化
    init_IRQ();//架构相关中断初始化
    tick_init();
    rcu_init_nohz();
    init_timers();//定时器初始化
    hrtimers_init();//高精度时钟初始化
    softirq_init(); //软中断初始化
    timekeeping_init(); // 初始化资源和普通计时器 
    time_init();//时间、定时器初始化(包括读取CMOS时钟、估测主频、初始化定时器中断等)
    sched_clock_postinit();
    perf_event_init();
    profile_init();// 对内核的一个性能测试工具profile进行初始化。
    call_function_init();
    WARN(!irqs_disabled(), "Interrupts were enabled early\n");
    early_boot_irqs_disabled = false;
    local_irq_enable();//使能中断 

    kmem_cache_init_late();//kmem_cache_init_late的目的就在于完善slab分配器的缓存机制.

    /*
     * HACK ALERT! This is early. We're enabling the console before
     * we've done PCI setups etc, and console_init() must be aware of
     * this. But we do want output early, in case something goes wrong.
     */
    console_init();//初始化控制台以显示printk的内容 
    if (panic_later)
        panic("Too many boot %s vars at `%s'", panic_later,
              panic_param);

    lockdep_info();// 如果定义了CONFIG_LOCKDEP宏,那么就打印锁依赖信息,否则什么也不做 

    /*
     * Need to run this when irqs are enabled, because it wants
     * to self-test [hard/soft]-irqs on/off lock inversion bugs
     * too:
     */
    locking_selftest();

#ifdef CONFIG_BLK_DEV_INITRD
    if (initrd_start && !initrd_below_start_ok &&
        page_to_pfn(virt_to_page((void *)initrd_start)) < min_low_pfn) {
        pr_crit("initrd overwritten (0x%08lx < 0x%08lx) - disabling it.\n",
            page_to_pfn(virt_to_page((void *)initrd_start)),
            min_low_pfn);
        initrd_start = 0;
    }
#endif
    page_cgroup_init();
    debug_objects_mem_init();
    kmemleak_init();
    setup_per_cpu_pageset();
    numa_policy_init();
    if (late_time_init)
        late_time_init();
    sched_clock_init();
    calibrate_delay();//延迟校准(获得时钟jiffies与CPU主频ticks的延迟)
    pidmap_init();//进程号位图初始化,一般用一个錺age来表示所有进程的錺id占用情况 
    anon_vma_init();// 匿名虚拟内存域( anonymous VMA)初始化
    acpi_early_init();
#ifdef CONFIG_X86
    if (efi_enabled(EFI_RUNTIME_SERVICES))
        efi_enter_virtual_mode();
#endif
#ifdef CONFIG_X86_ESPFIX64
    /* Should be run before the first non-init thread is created */
    init_espfix_bsp();
#endif
    thread_info_cache_init();//获取thread_info缓存空间,大部分构架为空函数(包括ARM )
    cred_init();//任务信用系统初始化。
    fork_init(totalram_pages);//进程创建机制初始化。为内核"task_struct"分配空间,计算最大任务数。
    proc_caches_init();//初始化进程创建机制所需的其他数据结构,为其申请空间。
    buffer_init();//块设备读写缓冲区初始化(同时创建"buffer_head"cache用户加速访问)
    key_init();//内核密钥管理系统初始化 
    security_init();//内核安全框架初始化
    dbg_late_init();
    vfs_caches_init(totalram_pages);//虚拟文件系统(VFS)缓存初始化 
    signals_init();//信号管理系统初始化
    /* rootfs populating might need page-writeback */
    page_writeback_init();//页写回机制初始化
    proc_root_init();//proc文件系统初始化
    cgroup_init();//control group正式初始化
    cpuset_init();//CPUSET初始化。
    taskstats_init_early();//任务状态早期初始化函数:为结构体获取高速缓存,并初始化互斥机制
    delayacct_init();//任务延迟初始化 

    check_bugs();//检查CPU BUG的函数,通过软件规避BUG 

    sfi_init_late();//功能跟踪调试机制初始化,ftrace 是 function trace 的简称 

    if (efi_enabled(EFI_RUNTIME_SERVICES)) {
        efi_late_init();
        efi_free_boot_services();
    }

    ftrace_init();

    /* Do the rest non-__init'ed, we're now alive */
    rest_init();  //rest_init()一旦启动就会创建0号进程作为idle进程,然后由0号进程创建一号进程(第一个用户态进程)并创建一个内核线程来管理系统资源及创建其他进程。
}

总结

Linux系统启动时先完成各个管理控制模块的初始化,然后创建0号进程即idle进程,该进程一直存在于系统中,当系统没有进程需要调度的时候就调度idle进程,1号进程和其他进程均由idle进程创建。

参考资料

http://blog.csdn.net/wzw12315/article/details/6304891
http://blog.csdn.net/dreamxu/article/details/6079664
http://blog.csdn.net/thegameisfives/article/details/7699625
http://blog.chinaunix.net/uid-20746260-id-3176497.html
http://www.it165.net/os/html/201409/9264.html



Sawoom原创作品转载请注明出处
《Linux内核分析》MOOC课程http://mooc.study.163.com/course/USTC-1000029000

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,616评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,020评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,078评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,040评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,154评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,265评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,298评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,072评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,491评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,795评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,970评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,654评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,272评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,985评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,815评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,852评论 2 351

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,913评论 25 707
  • 1 进程介绍 1.1 进程和程序 所谓进程是由正文段(text)、用户数据段(user segment)以及系统数...
    疯狂小王子阅读 1,229评论 0 7
  • Android中的异步任务机制 Android中AsyncTak的使用与源码分析 http://blog.csdn...
    Zaker2Magic阅读 514评论 0 6
  • 17岁懵懂的年纪,我们暧昧着。 爱玩的天性,我们远离彼此。 两年后再一次交集。原来命运是轮回着的,前世未尽的缘今生...
    o小巫女o阅读 271评论 0 0
  • 从77要来北京学习,就酝酿着因她而聚的狂欢。 开始定的聚会时间因为云燕要照顾住院的妈妈推后了一周,酝酿又加了一重等...
    真冉阅读 325评论 0 3