0

私はクラスのために小さなシェルを書いています。execvp組み込みでないコマンドを要求します。これを開発するのにしばらくの間、それは-1で正常に戻り、すべてが正しかった。ただし、シェルには何でも入力できます。シェルはプロセスを開始し、ステータス11(SIGSEGV、セグメンテーション違反)ですぐに終了します。

メタ:これをSSCCEに切り詰めようとしましたが、失敗しました。周囲のすべてを削除するとexecvp、正常に機能します。完全なソースを提供できてうれしいですが、750行を超えると、現時点では過剰に思えます。

execvpが呼び出される関数は次のとおりです。

void eval(char *cmdline)
{
char **argv = malloc(MAXLINE);
int bg = parseline(cmdline,argv);

if (argv[0] == 0)
{
    free(argv);
    return;
}
if(builtin_cmd(argv) == 0)
{
    pid_t pid;
    sigset_t set, oset;

    /* set mask */
    sigemptyset(&set);
    sigaddset(&set, SIGCHLD);
    sigprocmask(SIG_SETMASK, &set, &oset);

    /* proceed to fork */
    if ((pid = Fork()) == 0)
    {
        app_debug("execvp <%s>", argv[0]);
        setpgid(0,0);
        sigprocmask(SIG_UNBLOCK, &set, &oset);
        execvp(argv[0], argv);
        app_notify("%s command not found.", errno, argv[0]);
        exit(0);
    } else if (pid == -1) {
        app_notify("fork failed");
        return;
    } else {
        addjob(jobs, pid, bg + 1, cmdline);
        sigprocmask(SIG_UNBLOCK, &set, &oset);
        if (bg == 0) {
            app_debug("[%d] (%d) Waiting for %s", pid2jid(pid), pid, cmdline);
            waitfg(pid);
        } else {
            app_notify("[%d] (%d) %s", pid2jid(pid), pid, cmdline);
        }
    }
}
free(argv);
return;
}

これがparselineです。

int parseline(const char *cmdline, char **argv)
{
static char array[MAXLINE]; /* holds local copy of command line */
char *buf = array;          /* ptr that traverses command line */
char *delim;                /* points to first space delimiter */
int argc;                   /* number of args */
int bg;                     /* background job? */

strcpy(buf, cmdline);
buf[strlen(buf)-1] = ' ';  /* replace trailing '\n' with space */
while (*buf && (*buf == ' ')) /* ignore leading spaces */
buf++;

/* Build the argv list */
argc = 0;
if (*buf == '\'') {
buf++;
delim = strchr(buf, '\'');
}
else {
delim = strchr(buf, ' ');
}

while (delim) {
argv[argc++] = buf;
*delim = '\0';
buf = delim + 1;
while (*buf && (*buf == ' ')) /* ignore spaces */
       buf++;

if (*buf == '\'') {
    buf++;
    delim = strchr(buf, '\'');
}
else {
    delim = strchr(buf, ' ');
}
}
argv[argc] = NULL;

if (argc == 0)  /* ignore blank line */
return 1;

/* should the job run in the background? */
if ((bg = (*argv[argc-1] == '&')) != 0) {
argv[--argc] = NULL;
}
return bg;
}
4

1 に答える 1

0

execvpの直後に、app_notify("%s command not found.", errno, argv[0]);segに障害が発生しました。これは基本的にprintfをラップし、置換が1つしかないため、引用符で囲まれた文字列の後に引数を1つだけ持つ必要があります。

これをデバッグするための鍵はset follow-fork-mode child、gdbの検出でした。

あなたの助けをありがとう、ジム、等。

于 2012-10-08T22:48:11.760 に答える