linux内核剖析之main.c

main函数主要做一些初始化,比如内存,块设备、字符设备等的初始化,然后创建子进程打开bin/sh

 

    mem_init(main_memory_start,memory_end);
    trap_init();    // 陷阱门(硬件中断向量)初始化。(kernel/traps.c)
    blk_dev_init();    // 块设备初始化。(kernel/blk_dev/ll_rw_blk.c)
    chr_dev_init();    // 字符设备初始化。(kernel/chr_dev/tty_io.c)空,为以后扩展做准备。
    tty_init();        // tty 初始化。(kernel/chr_dev/tty_io.c)
    time_init();    // 设置开机启动时间 -> startup_time。
    sched_init();    // 调度程序初始化(加载了任务0 的tr, ldtr) (kernel/sched.c)
    buffer_init(buffer_memory_end);// 缓冲管理初始化,建内存链表等。(fs/buffer.c)
    hd_init();        // 硬盘初始化。(kernel/blk_dev/hd.c)
    floppy_init();    // 软驱初始化。(kernel/blk_dev/floppy.c)

 

 

在任务0中创建任务1

    if (!fork()) {        /* we count on this going ok */
        init();
    }

 

init函数:初始化shell环境,执行bin/sh
void init(void)
{
    int pid,i;

// 读取硬盘参数包括分区表信息并建立虚拟盘和安装根文件系统设备。
// 该函数是在25 行上的宏定义的,对应函数是sys_setup(),在kernel/blk_drv/hd.c。
    setup((void *) &drive_info);

    (void) open("/dev/tty0",O_RDWR,0);    // 用读写访问方式打开设备“/dev/tty0”,
                                        // 这里对应终端控制台。
                                        // 返回的句柄号0 -- stdin 标准输入设备。
    (void) dup(0);        // 复制句柄,产生句柄1 号-- stdout 标准输出设备。
    (void) dup(0);        // 复制句柄,产生句柄2 号-- stderr 标准出错输出设备。
    printf("%d buffers = %d bytes buffer space\n\r",NR_BUFFERS, \
        NR_BUFFERS*BLOCK_SIZE);    // 打印缓冲区块数和总字节数,每块1024 字节。
    printf("Free mem: %d bytes\n\r",memory_end-main_memory_start);//空闲内存字节数。

// 下面fork()用于创建一个子进程(子任务)。对于被创建的子进程,fork()将返回0 值,
// 对于原(父进程)将返回子进程的进程号。所以if (!(pid=fork())) {...} 内是子进程执行的内容。
// 该子进程关闭了句柄0(stdin),以只读方式打开/etc/rc 文件,并执行/bin/sh 程序,所带参数和
// 环境变量分别由argv_rc 和envp_rc 数组给出。参见后面的描述。
    if (!(pid=fork())) {
        close(0);
        if (open("/etc/rc",O_RDONLY,0))
            _exit(1);    // 如果打开文件失败,则退出(/lib/_exit.c)。
        execve("/bin/sh",argv_rc,envp_rc);    // 装入/bin/sh 程序并执行。(/lib/execve.c)
        _exit(2);    // 若execve()执行失败则退出(出错码2,“文件或目录不存在”)。
    }

// 下面是父进程执行的语句。wait()是等待子进程停止或终止,其返回值应是子进程的
// 进程号(pid)。这三句的作用是父进程等待子进程的结束。&i 是存放返回状态信息的
// 位置。如果wait()返回值不等于子进程号,则继续等待。
    if (pid>0)
        while (pid != wait(&i))
        {    /* nothing */;}

// --
// 如果执行到这里,说明刚创建的子进程的执行已停止或终止了。下面循环中首先再创建
// 一个子进程,如果出错,则显示“初始化程序创建子进程失败”的信息并继续执行。对
// 于所创建的子进程关闭所有以前还遗留的句柄(stdin, stdout, stderr),新创建一个
// 会话并设置进程组号,然后重新打开/dev/tty0 作为stdin,并复制成stdout 和stderr。
// 再次执行系统解释程序/bin/sh。但这次执行所选用的参数和环境数组另选了一套(见上面)。
// 然后父进程再次运行wait()等待。如果子进程又停止了执行,则在标准输出上显示出错信息
//        “子进程pid 停止了运行,返回码是i”,
// 然后继续重试下去…,形成“大”死循环。
    while (1) {
        if ((pid=fork())<0) {
            printf("Fork failed in init\r\n");
            continue;
        }
        if (!pid) {
            close(0);close(1);close(2);
            setsid();
            (void) open("/dev/tty0",O_RDWR,0);
            (void) dup(0);
            (void) dup(0);
            _exit(execve("/bin/sh",argv,envp));
        }
        while (1)
            if (pid == wait(&i))
                break;
        printf("\n\rchild %d died with code %04x\n\r",pid,i);
        sync();
    }
    _exit(0);    /* NOTE! _exit, not exit() */
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
第一章 走进linux 1.1 GNU与Linux的成长 1.2 Linux的开发模式和运作机制 1.3走进Linux内核 1.4 分析Linux内核的意义 1.5 Linux内核结构 1.6 Linux内核源代码 1.7 Linux内核源代码分析工具 第二章 Linux运行的硬件基础 2.1 i386的寄存器 2.2 内存地址 2.3 段机制和描述符 2.4 分页机制 2.5 Linux中的分页机制 2.6 Linux中的汇编语言 第三章中断机制 3.1 中断基本知识 3.2中断描述符表的初始化 3.3异常处理 3.4 中断处理 3.5中断的后半部分处理机制 第四章 进程描述 4.1 进程和程序(Process and Program) 4.2 Linux中的进程概述 4.3 task_struct结构描述 4.4 task_struct结构在内存中的存放 4.5 进程组织的方式 4.6 内核线程 4.7 进程的权能 4.8 内核同步 第五章进程调度 5.1 Linux时间系统 5.2 时钟中断 5.3 Linux的调度程序-Schedule( ) 5.4 进程切换 第六章 Linux内存管理 6.1 Linux的内存管理概述 6.2 Linux内存管理的初始化 6.3 内存的分配和回收 6.4 地址映射机制 6.5 请页机制 6.6 交换机制 6.7 缓存和刷新机制 6.8 进程的创建和执行 第七章 进程间通信 7.1 管道 7.2 信号(signal) 7.3 System V 的IPC机制 第八章 虚拟文件系统 8.1 概述 8.2 VFS中的数据结构 8.3 高速缓存 8.4 文件系统的注册、安装与拆卸 8.5 限额机制 8.6 具体文件系统举例 8.7 文件系统的系统调用 8 .8 Linux2.4文件系统的移植问题 第九章 Ext2文件系统 9.1 基本概念 9.2 Ext2的磁盘布局和数据结构 9.3 文件的访问权限和安全 9.4 链接文件 9.5 分配策略 第十章 模块机制 10.1 概述 10.2 实现机制 10.3 模块的装入和卸载 10.4 内核版本 10.5 编写内核模块 第十一章 设备驱动程序 11.1 概述 11.2 设备驱动基础 11.3 块设备驱动程序 11.4 字符设备驱动程序 第十二章 网络 12.1 概述 12.2 网络协议 12.3 套接字(socket) 12.4 套接字缓冲区(sk_buff) 12.5 网络设备接口 第十三章 启动系统 13.1 初始化流程 13.2 初始化的任务 13.3 Linux 的Boot Loarder 13.4 进入操作系统 13.5 main.c中的初始化 13.6 建立init进程 附录: 1 Linux 2.4内核API 2.1 驱动程序的基本函数 2.2 双向循环链表的操作 2.3 基本C库函数 2.4 Linux内存管理中Slab缓冲区 2.5 Linux中的VFS 2.6 Linux的连网 2.7 网络设备支持 2.8 模块支持 2.9 硬件接口 2.10 块设备 2.11 USB 设备
init/main.c 是 Linux 内核中的一个关键文件,它包含了 Linux 内核的主要初始化代码。该文件定义了内核的启动过程,包括初始化各个子系统、设备驱动和启动用户空间等。 具体分析 init/main.c 的代码需要查看具体版本的 Linux 内核代码。以下是一个简要的分析概述: 1. 初始化函数: - `start_kernel()`:这是 Linux 内核的入口函数,负责初始化内核数据结构、中断、内存管理、定时器等。 - `rest_init()`:该函数初始化系统的初始化任务(init task),并创建第一个用户进程(通常是 `/sbin/init`)。 2. 初始化子系统: - `kernel_init()`:该函数调用各个子系统的初始化函数,如进程管理(`sched_init()`)、内存管理(`mm_init()`)、文件系统(`vfs_caches_init()`)等。 3. 设备驱动初始化: - `drivers/base/init.c` 中的 `driver_init()` 函数:负责注册各个设备驱动。 - `drivers/char/tty_io.c` 中的 `tty_init()` 函数:初始化终端设备驱动。 4. 用户空间启动: - `kernel_init_freeable()`:该函数通过调用 `kernel_init()` 初始化用户空间,并将系统状态切换到用户模式。 总的来说,init/main.c 文件是 Linux 内核启动过程的关键文件,负责完成内核数据结构的初始化、子系统和设备驱动的初始化,最终将系统状态切换到用户空间。它是 Linux 内核启动过程中的重要一环。对于具体的代码细节,需要查看具体版本的 Linux 内核代码。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

jena_wy

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

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

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

打赏作者

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

抵扣说明:

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

余额充值