Chinaunix

标题: 想知道HelloWorld程序在Linux的从头到尾的运行流程该看哪些代码? [打印本页]

作者: guotong1988    时间: 2015-01-13 09:11
标题: 想知道HelloWorld程序在Linux的从头到尾的运行流程该看哪些代码?
比如一个print HelloWorld的程序
我想最初是gcc的事情吧?然后怎样和内核对接?内核又怎样怎样……
作者: 流氓无产者    时间: 2015-01-13 09:32
这个问题太大了
就像你提出生命如何诞生的一样
不同层次的人,回答不一样
作者: guotong1988    时间: 2015-01-13 09:37
回复 2# 流氓无产者


    先gcc hello.c -o hello

    再./hello

    重点是./hello

作者: guotong1988    时间: 2015-01-13 09:37
本帖最后由 guotong1988 于 2015-01-13 09:38 编辑

回复 2# 流氓无产者


    先gcc hello.c -o hello

    再./hello

  我想问的重点是./hello

作者: guotong1988    时间: 2015-01-13 09:58
先编译,编译就是把他转化成二进制指令。然后就是链接,链接就是与OS对接,加上一些协议数据,比如elf标识,形成一个Linux可以明白的可执行文件。 然后运行,Linux读取协议数据,将二进制指令载入内存,执行

作者: guotong1988    时间: 2015-01-13 10:15
具体和OS对接那块还是找不到看哪块代码啊。。。
作者: qxhgd    时间: 2015-01-13 11:27
搜下elf文件的执行原理即可。
作者: guotong1988    时间: 2015-01-14 14:42
int do_execve(char * filename,
        char __user *__user *argv,
        char __user *__user *envp,
        struct pt_regs * regs)
{
        struct linux_binprm *bprm;
        struct file *file;
        struct files_struct *displaced;
        bool clear_in_exec;
        int retval;

        retval = unshare_files(&displaced);
        if (retval)
                goto out_ret;

        retval = -ENOMEM;
        bprm = kzalloc(sizeof(*bprm), GFP_KERNEL);
        if (!bprm)
                goto out_files;

        retval = prepare_bprm_creds(bprm);
        if (retval)
                goto out_free;

        retval = check_unsafe_exec(bprm);
        if (retval < 0)
                goto out_free;
        clear_in_exec = retval;
        current->in_execve = 1;

        file = open_exec(filename);
        retval = PTR_ERR(file);
        if (IS_ERR(file))
                goto out_unmark;

        sched_exec();

        bprm->file = file;
        bprm->filename = filename;
        bprm->interp = filename;

        retval = bprm_mm_init(bprm);
        if (retval)
                goto out_file;

        bprm->argc = count(argv, MAX_ARG_STRINGS);
        if ((retval = bprm->argc) < 0)
                goto out;

        bprm->envc = count(envp, MAX_ARG_STRINGS);
        if ((retval = bprm->envc) < 0)
                goto out;

        retval = prepare_binprm(bprm);
        if (retval < 0)
                goto out;

        retval = copy_strings_kernel(1, &bprm->filename, bprm);
        if (retval < 0)
                goto out;

        bprm->exec = bprm->p;
        retval = copy_strings(bprm->envc, envp, bprm);
        if (retval < 0)
                goto out;

        retval = copy_strings(bprm->argc, argv, bprm);
        if (retval < 0)
                goto out;

        current->flags &= ~PF_KTHREAD;
        retval = search_binary_handler(bprm,regs);
        if (retval < 0)
                goto out;

        /* execve succeeded */
        current->fs->in_exec = 0;
        current->in_execve = 0;
        acct_update_integrals(current);
        free_bprm(bprm);
        if (displaced)
                put_files_struct(displaced);
        return retval;

out:
        if (bprm->mm) {
                acct_arg_size(bprm, 0);
                mmput(bprm->mm);
        }

out_file:
        if (bprm->file) {
                allow_write_access(bprm->file);
                fput(bprm->file);
        }

out_unmark:
        if (clear_in_exec)
                current->fs->in_exec = 0;
        current->in_execve = 0;

out_free:
        free_bprm(bprm);

out_files:
        if (displaced)
                reset_files_struct(displaced);
out_ret:
        return retval;
}





2.6的,看过之后也不知道怎么和fork联系起来的???
作者: super皮波    时间: 2015-01-14 15:07
回复 8# guotong1988

你在shell下运行的程序,shell进程会fork,然后在子进程中运行你的程序
   
作者: zsszss0000    时间: 2015-01-14 20:03
阅读《程序员的自我修养》回复 1# guotong1988


   
作者: Tinnal    时间: 2015-01-14 21:43
zsszss0000 发表于 2015-01-14 20:03
阅读《程序员的自我修养》回复 1# guotong1988


zsszss0000 是正解。这本书很好,满足你的需求。
如果你想看对应原源码,可以看binfmt_elf.c文件。编译器和操作系统的枢纽就是ELF格式。

ELF格式的文件头可以告诉内核在程序里有多少个段,每个段加载到什么地址,同时会调用程序指定的装载程序去装载程序(ld_linux.so)依赖的库。




欢迎光临 Chinaunix (http://bbs.chinaunix.net/) Powered by Discuz! X3.2