我在c。
中有关于execlp()的问题我有以下程序:
#include <stdio.h>
#include <unistd.h>
#include <sys/unistd.h>
#include <sys/types.h>
#include <pwd.h>
#include <stdlib.h>
#include <limits.h>
#include <signal.h>
void INThandler(int);
int main(int argc, char* argv[]) {
struct passwd *pwd;
char *lgn;
char *cwd;
char buff[PATH_MAX + 1];
char s1[10], s2[10];
/*Um den Namen zu bekommen*/
lgn = getlogin();
pwd = getpwnam(lgn);
/*Um den Hostnamen zu bekommen*/
char hostname[128];
gethostname(hostname, sizeof hostname);
/*Um das Arbeitsverzeichnis zu bekommen*/
cwd = getcwd(buff, PATH_MAX + 1);
if((cwd!=NULL)&& hostname!=NULL && ((lgn=getlogin())!=NULL ||
(pwd!=NULL)))
{
signal(SIGINT, INThandler);
while(1)
{
printf("%s@%s %s$", pwd->pw_name, hostname, cwd);
if(scanf("%s %s",s1, s2)<1)
return 1;
printf("Befehl: %s\nArgument: %s\n",s1,s2);
execlp(s1, s1, NULL);
printf("Zhopa");
return 1;
}
}
return 0;
}
void INThandler(int sig) {
char c;
signal(sig, SIG_IGN);
printf("Wollen Sie Program Verlassen? [y/n]");
c = getchar();
if(c == 'y' || c=='Y')
exit(0);
else
signal(SIGINT, INThandler);
getchar();
}
它应该打印用户名@ hostname文件夹$并将linux命令作为参数&#34; ls -al&#34; 之后它应该用execlp()启动它,但它并没有像我想的那样工作。
我在这里阅读了有关此命令的所有文章,但我想,我仍然不明白,如何使用它。
我很感激有人的帮助。
答案 0 :(得分:1)
您已使用fork()
创建新流程,然后在新流程(子)中使用execlp
。这是示例代码。它没有处理任何错误,它仅适用于具有1个参数的命令,因为这是我所理解的(例如ls -all
)
#include <stdio.h>
#include <unistd.h>
#include <sys/unistd.h>
#include <sys/wait.h> /*Lib for waitpid*/
#include <sys/types.h>
#include <pwd.h>
#include <stdlib.h>
#include <limits.h>
#include <signal.h>
void INThandler(int);
int main(int argc, char* argv[]) {
int pid = 0; /*PROCESS ID*/
struct passwd *pwd;
char *lgn;
char *cwd;
char buff[PATH_MAX + 1];
char s1[10], s2[10];
/*Um den Namen zu bekommen*/
lgn = getlogin();
pwd = getpwnam(lgn);
/*Um den Hostnamen zu bekommen*/
char hostname[128];
gethostname(hostname, sizeof hostname);
/*Um das Arbeitsverzeichnis zu bekommen*/
cwd = getcwd(buff, PATH_MAX + 1);
if((cwd!=NULL)&& hostname!=NULL && ((lgn=getlogin())!=NULL ||
(pwd!=NULL)))
{
signal(SIGINT, INThandler);
while(1)
{
printf("%s@%s %s$", pwd->pw_name, hostname, cwd);
if(scanf("%s %s",s1, s2)<1)
return 1;
printf("Befehl: %s\nArgument: %s\n",s1,s2);
pid = fork();
if(pid == 0){ /*Child*/
execlp(s1, s1, s2,(char*) NULL);
}else if(pid > 0){ /*Father*/
/*father waiting for the child*/
waitpid(pid,NULL,0);
}
printf("Zhopa");
}
}
return 0;
}
void INThandler(int sig) {
char c;
signal(sig, SIG_IGN);
printf("Wollen Sie Program Verlassen? [y/n]");
c = getchar();
if(c == 'y' || c=='Y')
exit(0);
else
signal(SIGINT, INThandler);
getchar();
}
答案 1 :(得分:1)
您的信号处理程序调用未定义的行为。
您只能在信号处理程序中调用异步信号安全函数。 Per the POSIX standard:
... 行为未定义 ...如果信号处理程序调用任何行为 除了其中一个功能之外,本标准中定义的功能 列于下表中。
下表定义了一组应该是的函数 异步信号安全。因此,应用程序可以调用它们,而无需 限制,来自信号捕捉功能。 ...
[异步信号安全功能表]
上表中没有的任何功能对于信号可能是不安全的。 ...
此信号处理程序
void INThandler(int sig) {
char c;
signal(sig, SIG_IGN);
printf("Wollen Sie Program Verlassen? [y/n]");
c = getchar();
if(c == 'y' || c=='Y')
exit(0);
else
signal(SIGINT, INThandler);
getchar();
}
中有多个非异步信号安全功能:
printf()
getchar()
exit()
无法从信号处理程序中安全地调用这些函数。
鉴于您的代码,调用此信号处理程序的可能位置是在scanf()
中阻止该进程时 - 尝试从stdin
读取。这可能涉及某种锁或互斥。然而你的信号处理程序调用{{1}},它也试图从getchar()
读取,这可能被锁定或处于某种不确定的状态。如果主线程在stdin
中被阻塞,则对信号处理程序中的scanf()
的异步调用可能会使用getchar()
的内部结构死锁或损坏。