[c/c++]6. exec函数

exec

image.png

曾提及用fork函数创建新的子进程后,子进程往往要调用一种exec函数以执行另一个程序。当进程调用一种exec函数时,该进程执行的程序完全替换为新程序,而新程序则从其main函数开始执行。因为调用exec不创建新进程,所以前后的进程ID并未改变。exec只是用磁盘 上的一个新程序替换了当前进程正文段数据段堆段栈段

有7种不同的exec函数可供使用,它们常常被统称为exec函数,我们 可以使用这7个函数中的任一个。这些exec函数使得UNIX系统进程控制 原语更加完善。用fork可以创建新进程,用exec可以初始执行新的程 序。exit函数和wait函数处理终止和等待终止。这些是我们需要的基本的 进程控制原语。

#include <unistd.h>
int execl(const char *pathname, const char *arg0, ... /* (char *)0 */ );

int execv(const char *pathname, char *const argv[]); 

int execle(const char *pathname, const char *arg0, .../* (char *)0, char *const envp[] */ );

int execve(const char *pathname, char *const argv[], char*const envp[]);

int execlp(const char *filename, const char *arg0, ... /*(char *)0 */ );

int execvp(const char *filename, char *const argv[]);
 
int fexecve(int fd, char *const argv[], char *const envp[]);

7个函数返回值:若出错,返回−1;若成功,不返回 这些函数之间的第一个区别是前4个函数取路径名作为参数,后两
个函数则取文件名作为参数,最后一个取文件描述符作为参数。当指定 filename作为参数时:

•如果filename中包含/,则就将其视为路径名;
•否则就按PATH环境变量,在它所指定的各目录中搜寻可执行文 件。
PATH 变量包含了一张目录表(称为路径前缀),目录之间用冒号(:)分隔。例如,下列name=value环境字符串指定在4个目录中进行搜 索。

PATH=/bin:/usr/bin:/usr/local/bin:.

最后的路径前缀.表示当前目录。(零长前缀也表示当前目录。在 value的开始处可用:表示,在行中间则要用::表示,在行尾以:表示。)

如果execlp或execvp使用路径前缀中的一个找到了一个可执行文件, 但是该文件不是由连接编辑器产生的机器可执行文件,则就认为该文件 是一个shell脚本,于是试着调用/bin/sh,并以该filename作为shell的输 入。

fexecve函数避免了寻找正确的可执行文件,而是依赖调用进程来完 成这项工作。调用进程可以使用文件描述符验证所需要的文件并且无竞 争地执行该文件。否则,拥有特权的恶意用户就可以在找到文件位置并 且验证之后,但在调用进程执行该文件之前替换可执行文件(或可执行 文件的部分路径)

函数 execl、execlp和execle要求将新程序的每个命令行参数都 说明为一个单独的参数。这种参数表以空指针结尾。对于另外4个函数 (execv、execvp、execve和fexecve),则应先构造一个指向各参数的指 针数组,然后将该数组地址作为这4个函数的参数。

在使用ISO C原型之前,对execl、execle和execlp三个函数表示命令 行参数的一般方法是:

char *arg0, char *arg1, ..., char *argn, (char *)0
这种语法显式地说明了最后一个命令行参数之后跟了一个空指针。 如果用常量0来表示一个空指针,则必须将它强制转换为一个指针;否 则它将被解释为整型参数。如果一个整型数的长度与char *的长度不 同,那么exec函数的实际参数将出错。

最后一个区别与向新程序传递环境表相关。以e结尾的3个函数 (execle、execve和fexecve)可以传递一个指向环境字符串指针数组的指 针。其他4个函数则使用调用进程中的environ变量为新程序复制现有的 环境(回忆7.9节及图7-8中对环境字符串的讨论。其中曾提及如果系统 支持setenv和putenv这样的函数,则可更改当前环境和后面生成的子进程 的环境,但不能影响父进程的环境)。通常,一个进程允许将其环境传 播给其子进程,但有时也有这种情况,进程想要为子进程指定某一个确 定的环境。例如,在初始化一个新登录的shell时,login程序通常创建一 个只定义少数几个变量的特殊环境,而在我们登录时,可以通过shell启 动文件,将其他变量加到环境中。
在使用ISO C原型之前,execle的参数是:

char *pathname, char *arg0, ..., char *argn, (char *)0, char *envp[]

从中可见,最后一个参数是指向环境字符串的各字符指针构成的数 组的指针。而在ISO C原型中,所有命令行参数、空指针和envp指针都 用省略号(...)表示。
这7个exec函数的参数很难记忆。函数名中的字符会给我们一些帮 助。字母p表示该函数取filename作为参数,并且用PATH环境变量寻找 可执行文件。字母l表示该函数取一个参数表,它与字母v互斥。v表示该 函数取一个argv[ ]矢量。最后,字母e表示该函数取envp[ ]数组,而不使 用当前环境。

image.png

每个系统对参数表和环境表的总长度都有一个限制。
这种限制是由ARG_MAX给出的。在POSIX.1系统中,此值至 少是4 096字节。当使用shell的文件名扩充功能产生一个文件名列表时, 可能会受到此值的限制。

为了摆脱对参数表长度的限制,我们可以使用xargs(1)命令,将长参 数表断开成几部分。

前面曾提及,在执行exec 后,进程ID没有改变。但新程序从调用进 程继承了的下列属性:

• 进程ID和父进程ID •实际用户ID和实际组ID •附属组ID
•进程组ID
•会话ID
•控制终端
•闹钟尚余留的时间
•当前工作目录
•根目录
•文件模式创建屏蔽字
•文件锁
•进程信号屏蔽
•未处理信号
•资源限制
•nice值
•tms_utime、tms_stime、tms_cutime以及tms_cstime值

注意文件描述符

对打开文件的处理与每个描述符的执行时关闭(close-on-exec)标
志值有关

FD_CLOEXEC标志的说明,进程 中每个打开描述符都有一个执行时关闭标志。若设置了此标志,则在执行exec 时关闭该描述符;否则该描述符仍打开。除非特地用fcntl设置了 该执行时关闭标志,否则系统的默认操作是在exec后仍保持这种描述符 打开。

意,在exec前后实际用户ID和实际组ID保持不变,而有效ID是否 改变则取决于所执行程序文件的设置用户ID位和设置组ID位是否设 置。如果新程序的设置用户ID位已设置,则有效用户ID变成程序文件 所有者的ID;否则有效用户ID不变。对组ID的处理方式与此相同。

在很多UNIX实现中,这7个函数中只有execve是内核的系统调用。 另外6个只是库函数,它们最终都要调用该系统调用。

image.png
#include <iostream>
#include <stdio.h>
#include <unistd.h>
#include <sys/wait.h>

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

推荐阅读更多精彩内容

  • 官网 中文版本 好的网站 Content-type: text/htmlBASH Section: User ...
    不排版阅读 4,380评论 0 5
  • Linux 进程管理与程序开发 进程是Linux事务管理的基本单元,所有的进程均拥有自己独立的处理环境和系统资源,...
    JamesPeng阅读 2,457评论 1 14
  • 史上最全的iOS面试题及答案 iOS面试小贴士———————————————回答好下面的足够了----------...
    Style_伟阅读 2,348评论 0 35
  • 立足东莞实际,通过借鉴、改造、整合、创新等方式,在中小学开展慕课教学改革,将线上慕课与线下班级授课有机结合,将人机...
    征空地带阅读 1,062评论 0 51
  • 北方的秋意愈浓,连着几日太空灰蒙蒙的。调皮的小孩可以肆意的玩闹,不再忌惮炙热的太阳。傍晚老人和小孩,占满了各种活动...
    文艺白的世界阅读 425评论 0 0