📄 exec.c
字号:
data_base -= PAGE_SIZE; if (page[i]) // 如果该页面存在, put_page (page[i], data_base); // 就放置该页面。 } return data_limit; // 最后返回数据段限长(64MB)。}/** 'do_execve()' executes a new program.*//** 'do_execve()'函数执行一个新程序。*///// execve()系统中断调用函数。加载并执行子进程(其它程序)。// 该函数系统中断调用(int 0x80)功能号__NR_execve 调用的函数。// 参数:eip - 指向堆栈中调用系统中断的程序代码指针eip 处,参见kernel/system_call.s 程序// 开始部分的说明;tmp - 系统中断调用本函数时的返回地址,无用;// filename - 被执行程序文件名;argv - 命令行参数指针数组;envp - 环境变量指针数组。// 返回:如果调用成功,则不返回;否则设置出错号,并返回-1。intdo_execve (unsigned long *eip, long tmp, char *filename, char **argv, char **envp){ struct m_inode *inode; // 内存中I 节点指针结构变量。 struct buffer_head *bh; // 高速缓存块头指针。 struct exec ex; // 执行文件头部数据结构变量。 unsigned long page[MAX_ARG_PAGES]; // 参数和环境字符串空间的页面指针数组。 int i, argc, envc; int e_uid, e_gid; // 有效用户id 和有效组id。 int retval; // 返回值。 int sh_bang = 0; // 控制是否需要执行脚本处理代码。// 参数和环境字符串空间中的偏移指针,初始化为指向该空间的最后一个长字处。 unsigned long p = PAGE_SIZE * MAX_ARG_PAGES - 4;// eip[1]中是原代码段寄存器cs,其中的选择符不可以是内核段选择符,也即内核不能调用本函数。 if ((0xffff & eip[1]) != 0x000f) panic ("execve called from supervisor mode");// 初始化参数和环境串空间的页面指针数组(表)。 for (i = 0; i < MAX_ARG_PAGES; i++) /* clear page-table */ page[i] = 0;// 取可执行文件的对应i 节点号。 if (!(inode = namei (filename))) /* get executables inode */ return -ENOENT;// 计算参数个数和环境变量个数。 argc = count (argv); envc = count (envp);// 执行文件必须是常规文件。若不是常规文件则置出错返回码,跳转到exec_error2(第347 行)。restart_interp: if (!S_ISREG (inode->i_mode)) { /* must be regular file */ retval = -EACCES; goto exec_error2; }// 检查被执行文件的执行权限。根据其属性(对应i 节点的uid 和gid),看本进程是否有权执行它。 i = inode->i_mode; e_uid = (i & S_ISUID) ? inode->i_uid : current->euid; e_gid = (i & S_ISGID) ? inode->i_gid : current->egid; if (current->euid == inode->i_uid) i >>= 6; else if (current->egid == inode->i_gid) i >>= 3; if (!(i & 1) && !((inode->i_mode & 0111) && suser ())) { retval = -ENOEXEC; goto exec_error2; }// 读取执行文件的第一块数据到高速缓冲区,若出错则置出错码,跳转到exec_error2 处去处理。 if (!(bh = bread (inode->i_dev, inode->i_zone[0]))) { retval = -EACCES; goto exec_error2; }// 下面对执行文件的头结构数据进行处理,首先让ex 指向执行头部分的数据结构。 ex = *((struct exec *) bh->b_data); /* read exec-header *//* 读取执行头部分 */// 如果执行文件开始的两个字节为'#!',并且sh_bang 标志没有置位,则处理脚本文件的执行。 if ((bh->b_data[0] == '#') && (bh->b_data[1] == '!') && (!sh_bang)) {/** This section does the #! interpretation.* Sorta complicated, but hopefully it will work. -TYT*//** 这部分处理对'#!'的解释,有些复杂,但希望能工作。-TYT*/ char buf[1023], *cp, *interp, *i_name, *i_arg; unsigned long old_fs;// 复制执行程序头一行字符'#!'后面的字符串到buf 中,其中含有脚本处理程序名。 strncpy (buf, bh->b_data + 2, 1022);// 释放高速缓冲块和该执行文件i 节点。 brelse (bh); iput (inode);// 取第一行内容,并删除开始的空格、制表符。 buf[1022] = '\0'; if (cp = strchr (buf, '\n')) { *cp = '\0'; for (cp = buf; (*cp == ' ') || (*cp == '\t'); cp++); }// 若该行没有其它内容,则出错。置出错码,跳转到exec_error1 处。 if (!cp || *cp == '\0') { retval = -ENOEXEC; /* No interpreter name found */ goto exec_error1; }// 否则就得到了开头是脚本解释执行程序名称的一行内容。 interp = i_name = cp;// 下面分析该行。首先取第一个字符串,其应该是脚本解释程序名,iname 指向该名称。 i_arg = 0; for (; *cp && (*cp != ' ') && (*cp != '\t'); cp++) { if (*cp == '/') i_name = cp + 1; }// 若文件名后还有字符,则应该是参数串,令i_arg 指向该串。 if (*cp) { *cp++ = '\0'; i_arg = cp; }/** OK, we've parsed out the interpreter name and* (optional) argument.*//** OK,我们已经解析出解释程序的文件名以及(可选的)参数。*/// 若sh_bang 标志没有设置,则设置它,并复制指定个数的环境变量串和参数串到参数和环境空间中。 if (sh_bang++ == 0) { p = copy_strings (envc, envp, page, p, 0); p = copy_strings (--argc, argv + 1, page, p, 0); }/** Splice in (1) the interpreter's name for argv[0]* (2) (optional) argument to interpreter* (3) filename of shell script** This is done in reverse order, because of how the* user environment and arguments are stored.*//** 拼接 (1) argv[0]中放解释程序的名称* (2) (可选的)解释程序的参数* (3) 脚本程序的名称** 这是以逆序进行处理的,是由于用户环境和参数的存放方式造成的。*/// 复制脚本程序文件名到参数和环境空间中。 p = copy_strings (1, &filename, page, p, 1);// 复制解释程序的参数到参数和环境空间中。 argc++; if (i_arg) { p = copy_strings (1, &i_arg, page, p, 2); argc++; }// 复制解释程序文件名到参数和环境空间中。若出错,则置出错码,跳转到exec_error1。 p = copy_strings (1, &i_name, page, p, 2); argc++; if (!p) { retval = -ENOMEM; goto exec_error1; }/** OK, now restart the process with the interpreter's inode.*//** OK,现在使用解释程序的i 节点重启进程。*/// 保留原fs 段寄存器(原指向用户数据段),现置其指向内核数据段。 old_fs = get_fs (); set_fs (get_ds ());// 取解释程序的i 节点,并跳转到restart_interp 处重新处理。 if (!(inode = namei (interp))) { /* get executables inode */ set_fs (old_fs); retval = -ENOENT; goto exec_error1; } set_fs (old_fs); goto restart_interp; }// 释放该缓冲区。 brelse (bh);// 下面对执行头信息进行处理。// 对于下列情况,将不执行程序:如果执行文件不是需求页可执行文件(ZMAGIC)、或者代码重定位部分// 长度a_trsize 不等于0、或者数据重定位信息长度不等于0、或者代码段+数据段+堆段长度超过50MB、// 或者i 节点表明的该执行文件长度小于代码段+数据段+符号表长度+执行头部分长度的总和。 if (N_MAGIC (ex) != ZMAGIC || ex.a_trsize || ex.a_drsize || ex.a_text + ex.a_data + ex.a_bss > 0x3000000 || inode->i_size < ex.a_text + ex.a_data + ex.a_syms + N_TXTOFF (ex)) { retval = -ENOEXEC; goto exec_error2; }// 如果执行文件执行头部分长度不等于一个内存块大小(1024 字节),也不能执行。转exec_error2。 if (N_TXTOFF (ex) != BLOCK_SIZE) { printk ("%s: N_TXTOFF != BLOCK_SIZE. See a.out.h.", filename); retval = -ENOEXEC; goto exec_error2; }// 如果sh_bang 标志没有设置,则复制指定个数的环境变量字符串和参数到参数和环境空间中。// 若sh_bang 标志已经设置,则表明是将运行脚本程序,此时环境变量页面已经复制,无须再复制。 if (!sh_bang) { p = copy_strings (envc, envp, page, p, 0); p = copy_strings (argc, argv, page, p, 0);// 如果p=0,则表示环境变量与参数空间页面已经被占满,容纳不下了。转至出错处理处。 if (!p) { retval = -ENOMEM; goto exec_error2; } }/* OK, This is the point of no return *//* OK,下面开始就没有返回的地方了 */// 如果原程序也是一个执行程序,则释放其i 节点,并让进程executable 字段指向新程序i 节点。 if (current->executable) iput (current->executable); current->executable = inode;// 清复位所有信号处理句柄。但对于SIG_IGN 句柄不能复位,因此在322 与323 行之间需添加一条// if 语句:if (current->sa[I].sa_handler != SIG_IGN)。这是源代码中的一个bug。 for (i = 0; i < 32; i++) current->sigaction[i].sa_handler = NULL;// 根据执行时关闭(close_on_exec)文件句柄位图标志,关闭指定的打开文件,并复位该标志。 for (i = 0; i < NR_OPEN; i++) if ((current->close_on_exec >> i) & 1) sys_close (i); current->close_on_exec = 0;// 根据指定的基地址和限长,释放原来程序代码段和数据段所对应的内存页表指定的内存块及页表本身。 free_page_tables (get_base (current->ldt[1]), get_limit (0x0f)); free_page_tables (get_base (current->ldt[2]), get_limit (0x17));// 如果“上次任务使用了协处理器”指向的是当前进程,则将其置空,并复位使用了协处理器的标志。 if (last_task_used_math == current) last_task_used_math = NULL; current->used_math = 0;// 根据a_text 修改局部表中描述符基址和段限长,并将参数和环境空间页面放置在数据段末端。// 执行下面语句之后,p 此时是以数据段起始处为原点的偏移值,仍指向参数和环境空间数据开始处,// 也即转换成为堆栈的指针。 p += change_ldt (ex.a_text, page) - MAX_ARG_PAGES * PAGE_SIZE;// create_tables()在新用户堆栈中创建环境和参数变量指针表,并返回该堆栈指针。 p = (unsigned long) create_tables ((char *) p, argc, envc);// 修改当前进程各字段为新执行程序的信息。令进程代码段尾值字段end_code = a_text;令进程数据// 段尾字段end_data = a_data + a_text;令进程堆结尾字段brk = a_text + a_data + a_bss。 current->brk = ex.a_bss + (current->end_data = ex.a_data + (current->end_code = ex.a_text));// 设置进程堆栈开始字段为堆栈指针所在的页面,并重新设置进程的用户id 和组id。 current->start_stack = p & 0xfffff000; current->euid = e_uid; current->egid = e_gid;// 初始化一页bss 段数据,全为零。 i = ex.a_text + ex.a_data; while (i & 0xfff) put_fs_byte (0, (char *) (i++));// 将原调用系统中断的程序在堆栈上的代码指针替换为指向新执行程序的入口点,并将堆栈指针替换// 为新执行程序的堆栈指针。返回指令将弹出这些堆栈数据并使得CPU 去执行新的执行程序,因此不会// 返回到原调用系统中断的程序中去了。 eip[0] = ex.a_entry; /* eip, magic happens :-) *//* eip,魔法起作用了 */ eip[3] = p; /* stack pointer *//* esp,堆栈指针 */ return 0;exec_error2: iput (inode);exec_error1: for (i = 0; i < MAX_ARG_PAGES; i++) free_page (page[i]); return (retval);}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -