迷茫2017-04-17 15:32:43
謝答。
但是各位答主都沒有給出正確答案,這怪我資訊沒有給全。
其實程式碼本身沒有任何問題,但是前面還有一段setrlimit
以限制記憶體使用,由於配置失誤,記憶體上限過小,導致甚至無法存放argv
中的字串。
錯誤很狗血。
錯誤很狗血。
錯誤很狗血。
但至少我們知道了argv
是佔用目前進程記憶體的不是。 :P
高洛峰2017-04-17 15:32:43
看下面的程式碼
int main(int c,char** v)
{
v[c-1] = NULL;
execvp("ls",v);
perror("execvp");
}
編譯後運行,輸出如下
> ./runner / /
bin boot dev etc home lib lib64 lost+found mnt opt proc root run sbin srv sys tmp usr var
修改一下程式碼,讓execvp
的參數不已NULL
結尾
int main(int c,char** v)
{
v[c] = "123";
execvp("ls",v);
perror("execvp");
}
編譯後執行
> ./runner
./runner: 无法访问'123': 没有那个文件或目录
./runner: 无法访问'USER=o': 没有那个文件或目录
./runner: 无法访问'LOGNAME=o': 没有那个文件或目录
./runner: 无法访问'HOME=/home/o': 没有那个文件或目录
./runner: 无法访问'PATH=/usr/local/sbin:/usr/local/bin:/usr/bin:/usr/bin/site_perl:/usr/bin/vendor_perl:/usr/bin/core_perl:/home/o/program/open64-5.0/bin': 没有那个文件或目录
./runner: 无法访问'MAIL=/var/spool/mail/o': 没有那个文件或目录
./runner: 无法访问'SHELL=/usr/bin/zsh': 没有那个文件或目录
./runner: 无法访问'SSH_CLIENT=192.168.0.25 22975 22': 没有那个文件或目录
./runner: 无法访问'SSH_CONNECTION=192.168.0.25 22975 192.168.0.241 22': 没有那个文件或目录
./runner: 无法访问'SSH_TTY=/dev/pts/0': 没有那个文件或目录
./runner: 无法访问'TERM=xterm': 没有那个文件或目录
./runner: 无法访问'XDG_SESSION_ID=c6': 没有那个文件或目录
./runner: 无法访问'XDG_RUNTIME_DIR=/run/user/1000': 没有那个文件或目录
./runner: 无法访问'DBUS_SESSION_BUS_ADDRESS=unix:path=/run/user/1000/bus': 没有那个文件或目录
./runner: 无法访问'SHLVL=1': 没有那个文件或目录
./runner: 无法访问'PWD=/home/o/wTest/src/linux': 没有那个文件或目录
./runner: 无法访问'OLDPWD=/home/o/wTest/src': 没有那个文件或目录
./runner: 无法访问'LANG=en_US.UTF-8': 没有那个文件或目录
./runner: 无法访问'ZSH=/home/o/.oh-my-zsh': 没有那个文件或目录
./runner: 无法访问'LC_ALL=zh_CN.UTF-8': 没有那个文件或目录
./runner: 无法访问'LS_COLORS=rs=0:di=01;34:ln=01;36:mh=00:pi=40;33:so=01;35:do=01;35:bd=40;33;01:cd=40;33;01:or=40;31;01:mi=00:su=37;41:sg=30;43:ca=30;41:tw=30;42:ow=34;42:st=37;44:ex=01;32:*.tar=01;31:*.tgz=01;31:*.arc=01;31:*.arj=01;31:*.taz=01;31:*.lha=01;31:*.lz4=01;31:*.lzh=01;31:*.lzma=01;31:*.tlz=01;31:*.txz=01;31:*.tzo=01;31:*.t7z=01;31:*.zip=01;31:*.z=01;31:*.Z=01;31:*.dz=01;31:*.gz=01;31:*.lrz=01;31:*.lz=01;31:*.lzo=01;31:*.xz=01;31:*.bz2=01;31:*.bz=01;31:*.tbz=01;31:*.tbz2=01;31:*.tz=01;31:*.deb=01;31:*.rpm=01;31:*.jar=01;31:*.war=01;31:*.ear=01;31:*.sar=01;31:*.rar=01;31:*.alz=01;31:*.ace=01;31:*.zoo=01;31:*.cpio=01;31:*.7z=01;31:*.rz=01;31:*.cab=01;31:*.jpg=01;35:*.jpeg=01;35:*.gif=01;35:*.bmp=01;35:*.pbm=01;35:*.pgm=01;35:*.ppm=01;35:*.tga=01;35:*.xbm=01;35:*.xpm=01;35:*.tif=01;35:*.tiff=01;35:*.png=01;35:*.svg=01;35:*.svgz=01;35:*.mng=01;35:*.pcx=01;35:*.mov=01;35:*.mpg=01;35:*.mpeg=01;35:*.m2v=01;35:*.mkv=01;35:*.webm=01;35:*.ogm=01;35:*.mp4=01;35:*.m4v=01;35:*.mp4v=01;35:*.vob=01;35:*.qt=01;35:*.nuv=01;35:*.wmv=01;35:*.asf=01;35:*.rm=01;35:*.rmvb=01;35:*.flc=01;35:*.avi=01;35:*.fli=01;35:*.flv=01;35:*.gl=01;35:*.dl=01;35:*.xcf=01;35:*.xwd=01;35:*.yuv=01;35:*.cgm=01;35:*.emf=01;35:*.ogv=01;35:*.ogx=01;35:*.aac=00;36:*.au=00;36:*.flac=00;36:*.m4a=00;36:*.mid=00;36:*.midi=00;36:*.mka=00;36:*.mp3=00;36:*.mpc=00;36:*.ogg=00;36:*.ra=00;36:*.wav=00;36:*.oga=00;36:*.opus=00;36:*.spx=00;36:*.xspf=00;36:': 文件名过长
./runner: 无法访问'PAGER=less': 没有那个文件或目录
./runner: 无法访问'LESS=-R': 没有那个文件或目录
./runner: 无法访问'LSCOLORS=Gxfxcxdxbxegedabagacad': 没有那个文件或目录
./runner: 无法访问'_=/home/o/wTest/src/linux/./runner': 没有那个文件或目录
可以看到,這裡把目前的環境變數輸出了。
你的config->args裡面如果沒有以NULL
結尾,而你的環境變數又很多的話,是不是就會這樣了呢?
沒有詳細讀你的程式碼,只是簡單的看了一下execvp
呼叫的部分。 你這裡沒有判斷execvp
是否成功,就直接呼叫了raise
函數。 但是raise
這個函數用來向正在執行的程式發送一個訊號。也就是說你這裡只要呼叫了execvp
,不出意外就會發送SIGUSR1
訊號的。 因為errno
這是一個全域的變量,所以在這裡,它並不能說明其是execvp
設定的。
ringa_lee2017-04-17 15:32:43
看起來像是 config->args
指標數組最後一個元素沒有設定為 NULL
。
execvp
是根據空指標來判斷列表結尾的。
例如:
config->path = "/path/to/your/bin";
config->args = {
"abc",
"cde",
NULL // 如果没有加上这个,就会出现你说的错误
};
execvp(config->path, (char * const*)config->args);