在c程序中使用execlp()

时间:2017-05-14 10:32:00

标签: c execl

我在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()启动它,但它并没有像我想的那样工作。

我在这里阅读了有关此命令的所有文章,但我想,我仍然不明白,如何使用它。

我很感激有人的帮助。

2 个答案:

答案 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()的内部结构死锁或损坏。