搜尋

首頁  >  問答  >  主體

linux下execvp失败设置errno=7的原因是什么?

伊谢尔伦伊谢尔伦2786 天前748

全部回覆(4)我來回復

  • 迷茫

    迷茫2017-04-17 15:32:43

    謝答。
    但是各位答主都沒有給出正確答案,這怪我資訊沒有給全。
    其實程式碼本身沒有任何問題,但是前面還有一段setrlimit以限制記憶體使用,由於配置失誤,記憶體上限過小,導致甚至無法存放argv中的字串。
    錯誤很狗血。
    錯誤很狗血。
    錯誤很狗血。
    但至少我們知道了argv是佔用目前進程記憶體的不是。 :P

    回覆
    0
  • 高洛峰

    高洛峰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設定的。

    回覆
    0
  • PHPz

    PHPz2017-04-17 15:32:43

    把ab放後台,如果還有錯誤那就一定是ab執行的錯誤

    回覆
    0
  • ringa_lee

    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);

    回覆
    0
  • 取消回覆