Я запускаю эту основную программу оболочки в другой оболочке. Я не могу понять, почему моя оболочка не работает после выполнения «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);
}
за пределами инструкции if? – user2125805
Да, я был бы склонен к 'fork()' перед 'if', а не forking для каждого из случаев' if' отдельно - DRY (не повторяйте себя) - хороший девиз. – Simon