我正在为课堂写一个小贝壳。我为非内置命令调用execvp
。在开发它的过程中,它通常返回-1,一切都是正确的。但是,现在,我可以在shell中键入任何内容,它会启动一个立即以状态11终止的进程(SIGSEGV,一个分段错误)。
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;
}
答案 0 :(得分:0)
在execvp之后,app_notify("%s command not found.", errno, argv[0]);
seg出现故障。它基本上包装了一个printf,并且在引用的字符串后面应该只有一个参数,因为只有一个替换。
调试此问题的关键是为gdb发现set follow-fork-mode child
。
感谢您的帮助,Jim等人