私はクラスのために小さなシェルを書いています。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;
}