2013-03-06 2 views
1

Я запускаю эту основную программу оболочки в другой оболочке. Я не могу понять, почему моя оболочка не работает после выполнения «ls». У меня нет выхода, но он возвращается к исходной оболочке. Я должен запускать свою программу оболочки каждый раз, если вы хотите ее использовать. Я понял, что должен делать fork(). Я только хочу, чтобы моя оболочка завершила работу с помощью команды exit, которую я закодировал с помощью инструкции if else. Любые предложения будут высоко ценится. О, и не обращайте внимание на функцию парсера gettoks(), я не мог понять, как использовать ее для ввода, поэтому я написал, если инструкции else для строкового ввода cmSTR, а не используя парсер gettoks(). Главным образом потому, что я не мог понять, как пройти вход в негоКак сохранить мою оболочку после команды

#include <iostream> 
#include <unistd.h> 
#include <sys/types.h> 
#include <errno.h> 
#include <signal.h> 
#include <cstdlib> 
#include <sys/wait.h> 

using namespace std; 
// Initializing counters for trapping 
static int cc_counter = 0; 
static int cz_counter = 0; 
static int cq_counter = 0; 
//Functions for trap signal handler 
void cc_handler(int signo) 
{ 
++cc_counter; 
} 
void cz_handler(int signo) 
{ 
++cz_counter; 
} 
void cq_handler(int signo) 
{ 
++cq_counter; 
} 

//********************************************************* 
// 
// Extern Declarations 
// 
//********************************************************* 
using namespace std; 
extern "C" 
{ 
extern char **gettoks(); 
} 


//********************************************************* 
// 
// Main Function 
// 
//********************************************************* 
int main(int argc, char *argv[]) 
{ 
// local variables 
int ii; 
char **toks; 
int retval; 

// initialize local variables 
ii = 0; 
toks = NULL; 
retval = 0; 
char buf[1000];//Initialize of size for current working directory 
string cmSTR;//String to hold input 
int status;//Initialization of status for fork() 
pid_t pid;//Declaration of pid 

// main (infinite) loop 
while(true) 
{ 
    signal(SIGINT, cc_handler);// Traps Ctrl+C 
    signal(SIGTSTP, cz_handler);// Traps Ctrl+Z 
    signal(SIGQUIT, cq_handler);// Traps Ctrl+\ 
    //prompt and show current working directory 
    cout <<("RS_SHELL:") << getcwd(buf,1000) << "\t"; 
    getline(cin ,cmSTR);//read input from keyboard 
    // if else loop to switch based on command input 
    if(cmSTR == "ls")// if ls, then execute arguement 
    { 
    execl("/bin/ls", "ls", NULL);//System call to execute ls 

    } 
    else if(cmSTR == "exit")//if exit, then execute block of code 
    { 
     cout << "Ctrl C entered: " << ++cc_counter << "times"<< endl; 
     cout << "Ctrl Z entered: " << ++cz_counter << "times"<< endl; 
     cout << "Ctrl Back Slash entered: " << ++cq_counter << "times"<< endl; 
     exit(1); 
     } 
    else if(cmSTR == "guish")// if guish, execute guish shell 
    { 
     execvp("guish", NULL); 
     } 
     //if input is not any of previous commands then fork() 
    else if(cmSTR != "ls" && cmSTR != "exit" && cmSTR != "guish" && cmSTR != "\n") 
    { 
     pid = fork(); 
     if (pid < 0)//Loop to fork parent and child process 
     { 
      fprintf(stderr, "Fork Failed"); 
      exit(-1); 
      } 
      else if (pid == 0)//Child process 
     { 
      execvp("guish", NULL);//system call to execute guish shell 

      } 
      else //Parent process 
      { 
       waitpid(-1, &status,0); 
       exit(0); 
       } 
       } 




    // get arguments 
    toks = gettoks(); 

    if(toks[0] != NULL) 
{ 
    // simple loop to echo all arguments 
    for(ii=0; toks[ii] != NULL; ii++) 
    { 
     cout << "Argument " << ii << ": " << toks[ii] << endl; 
    } 

    if(!strcmp(toks[0], "exit")) 
    break; 
} 
    } 

    // return to calling environment 
    return(retval); 
    } 

ответ

0

Как вы подозревали, execl и связанные с ним функции наложения текущего процесса с новым процессом. Таким образом, после вызова execl, который начинается с ls, ваша программа больше не будет существовать для продолжения работы.

Если вы хотите, чтобы ваша программа-оболочка, чтобы остаться вокруг после запуска ls, вам необходимо fork()перед тем на вызов execl("/bin/ls", "ls", NULL);.

Кроме того, если вы хотите, чтобы вывод из ls отображался в той же консоли, что и ваша оболочка, как я думаю, вы можете намереваться, вам нужно будет вывести вывод из ls обратно в вашу оболочку, а затем записать этот вывод на консоли вашего корпуса. См., Например, Writing my own shell… stuck on pipes?.

+0

за пределами инструкции if? – user2125805

+0

Да, я был бы склонен к 'fork()' перед 'if', а не forking для каждого из случаев' if' отдельно - DRY (не повторяйте себя) - хороший девиз. – Simon

Смежные вопросы